图片压缩算法机制—近微信压缩机制的Luban

这是项目地址

目前做App开发总绕不开图片这个元素。但是随着手机拍照分辨率的提升,图片的压缩成为一个很重要的问题。单纯对图片进行裁切,压缩已经有很多文章介绍。但是裁切成多少,压缩成多少却很难控制好,裁切过头图片太小,质量压缩过头则显示效果太差。

效果图


压缩比例

于是自然想到App巨头“微信”会是怎么处理,Luban(鲁班)就是通过在微信朋友圈发送近100张不同分辨率图片,对比原图与微信压缩后的图片逆向推算出来的压缩算法。

因为有其他语言也想要实现Luban,所以描述了一遍算法步骤。

因为是逆向推算,效果还没法跟微信一模一样,但是已经很接近微信朋友圈压缩后的效果,具体看以下对比!

效果与对比

内容原图LubanWechat
截屏720P 720*1280,390k720*1280,87k720*1280,56k
截屏1080P1080*1920,2.21M1080*1920,104k
拍照13M(4:3)3096*4128,3.12M1548*2064,141k
拍照9.6M(16:9)4128*2322,4.64M1032*581,97k
滚动截屏1080*6433,1.56M1080*6433,351k1080*6433,482k

使用方法

  • 第一步导入


compile 'top.zibin:Luban:1.1.3'
  • 异步调用

    Luban内部采用IO线程进行图片压缩,外部调用只需设置好结果监听即可:

    Luban.with(this)        
            .load(photos) // 传人要压缩的图片列表
            .ignoreBy(100)// 忽略不压缩图片的大小
            .setTargetDir(getPath()) // 设置压缩后文件存储位置
            .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(photos)
        .observeOn(Schedulers.io())
        .map(new Function<List<String>, List<File>>() {      
            @Override 
            public List<File> apply(@NonNull List<String> list) throws Exception {        // 同步方法直接返回压缩后的文件
                return Luban.with(MainActivity.this).load(list).get();
          }
        })
        .observeOn(AndroidSchedulers.mainThread())
        .subscribe();

项目地址:

https://github.com/Curzibn/Luban

终端研发部提倡: 没有做不到的,只有想不到的