1. 程式人生 > >android圖片壓縮的兩個開源庫

android圖片壓縮的兩個開源庫

Luban(魯班) —— Android圖片壓縮工具,仿微信朋友圈壓縮策略。

專案描述

目前做App開發總繞不開圖片這個元素。但是隨著手機拍照解析度的提升,圖片的壓縮成為一個很重要的問題。單純對圖片進行裁切,壓縮已經有很多文章介紹。但是裁切成多少,壓縮成多少卻很難控制好,裁切過頭圖片太小,質量壓縮過頭則顯示效果太差。

於是自然想到App巨頭“微信”會是怎麼處理,Luban(魯班)就是通過在微信朋友圈傳送近100張不同解析度圖片,對比原圖與微信壓縮後的圖片逆向推算出來的壓縮演算法。

因為有其他語言也想要實現Luban,所以描述了一遍演算法步驟

因為是逆向推算,效果還沒法跟微信一模一樣,但是已經很接近微信朋友圈壓縮後的效果,具體看以下對比!

效果與對比

內容 原圖 Luban Wechat
截圖 720P 720*1280,390k 720*1280,87k 720*1280,56k
截圖 1080P 1080*1920,2.21M 1080*1920,104k 1080*1920,112k
拍照 13M(4:3) 3096*4128,3.12M 1548*2064,141k 1548*2064,147k
拍照 9.6M(16:9) 4128*2322,4.64M 1032*581,97k 1032*581,74k
滾動截圖 1080*6433,1.56M 1080*6433,351k 1080*6433,482k

匯入

compile 'top.zibin:Luban:1.1.2'

使用

非同步呼叫

Luban內部採用IO執行緒進行圖片壓縮,外部呼叫只需設定好結果監聽即可:

Luban.with(this)
    .load(File)                     //傳人要壓縮的圖片
    .setCompressListener(new OnCompressListener() { //設定回撥
        @Override
        public void onStart() {
            // TODO 壓縮開始前呼叫,可以在方法內啟動 loading UI
        }
        @Override
        public
void onSuccess(File file) { // TODO 壓縮成功後呼叫,返回壓縮後的圖片檔案 } @Override public void onError(Throwable e) { // TODO 當壓縮過程出現問題時呼叫 } }).launch(); //啟動壓縮

同步呼叫

同步方法請儘量避免在主執行緒呼叫以免阻塞主執行緒,下面以rxJava呼叫為例

Flowable.just(file)
    .observeOn(Schedulers.io())
    .map(new Function<File, File>() {
      @Override public File apply(@NonNull File file) throws Exception {
        // 同步方法直接返回壓縮後的檔案
        return Luban.with(MainActivity.this).load(file).get();
      }
    })
    .observeOn(AndroidSchedulers.mainThread())
    .subscribe();

compressor

Gradle

dependencies {
    compile 'id.zelory:compressor:2.0.0'
}

Let's compress the image size!

Compress Image File

compressedImageFile = new Compressor(this).compressToFile(actualImageFile);

Compress Image File to Bitmap

compressedImageBitmap = new Compressor(this).compressToBitmap(actualImageFile);

I want custom Compressor!

compressedImage = new Compressor(this)
            .setMaxWidth(640)
            .setMaxHeight(480)
            .setQuality(75)
            .setCompressFormat(Bitmap.CompressFormat.WEBP)
            .setDestinationDirectoryPath(Environment.getExternalStoragePublicDirectory(
              Environment.DIRECTORY_PICTURES).getAbsolutePath())
            .compressToFile(actualImage);

Stay cool compress image asynchronously with RxJava!

new Compressor(this)
        .compressToFileAsFlowable(actualImage)
        .subscribeOn(Schedulers.io())
        .observeOn(AndroidSchedulers.mainThread())
        .subscribe(new Consumer<File>() {
            @Override
            public void accept(File file) {
                compressedImage = file;
            }
        }, new Consumer<Throwable>() {
            @Override
            public void accept(Throwable throwable) {
                throwable.printStackTrace();
                showError(throwable.getMessage());
            }
        });