Android开源 - Luban(鲁班)可能是最接近微信朋友圈的图片压缩算法

syicross 8年前
   <h2>前言</h2>    <p>Luban 是图片压缩工具,通过参考或者自创压缩规则推求极致的压缩效果 目前的版本压缩效果主要参考微信。因为微信用户量最大,如果压缩后的图片越接近微信则越被用户接受。</p>    <h2>说明</h2>    <p>目前的 Luban 只是压缩结果接近微信,自身的算法只是为了达到这个效果而设计的。与微信并无任何联系,也不敢妄称是微信的算法。</p>    <h2>算法步骤</h2>    <p>注:下文所说“比例”统一表示:图片短边除以长边为该图片比例</p>    <h2>第三挡压缩(参考最新版微信压缩效果)</h2>    <ol>     <li> <p>判断图片比例值,是否处于以下区间内;</p>      <ul>       <li>[1, 0.5625) 即图片处于 [1:1 ~ 9:16) 比例范围内</li>       <li>[0.5625, 0.5) 即图片处于 [9:16 ~ 1:2) 比例范围内</li>       <li>[0.5, 0) 即图片处于 [1:2 ~ 1:∞) 比例范围内</li>      </ul> </li>     <li> <p>判断图片最长边是否过边界值;</p>      <ul>       <li>[1, 0.5625) 边界值为:1664 * n(n=1), 4990 * n(n=2), 1280 * pow(2, n-1)(n≥3)</li>       <li>[0.5625, 0.5) 边界值为:1280 * pow(2, n-1)(n≥1)</li>       <li>[0.5, 0) 边界值为:1280 * pow(2, n-1)(n≥1)</li>      </ul> </li>     <li> <p>计算压缩图片实际边长值,以第2步计算结果为准,超过某个边界值则:width / pow(2, n-1),height/pow(2, n-1)</p> </li>     <li> <p>计算压缩图片的实际文件大小,以第2、3步结果为准,图片比例越大则文件越大。</p> <p>size = (newW * newH) / (width * height) * m;</p>      <ul>       <li>[1, 0.5625) 则 width & height 对应 1664,4990,1280 * n(n≥3),m 对应 150,300,300;</li>       <li>[0.5625, 0.5) 则 width = 1440,height = 2560, m = 200;</li>       <li>[0.5, 0) 则 width = 1280,height = 1280 / scale,m = 500;注:scale为比例值</li>      </ul> </li>     <li> <p>判断第4步的size是否过小</p>      <ul>       <li>[1, 0.5625) 则最小 size 对应 60,60,100</li>       <li>[0.5625, 0.5) 则最小 size 都为 100</li>       <li>[0.5, 0) 则最小 size 都为 100</li>      </ul> </li>     <li>将前面求到的值压缩图片 width, height, size 传入压缩流程,压缩图片直到满足以上数值</li>    </ol>    <h2>效果与对比</h2>    <table>     <thead>      <tr>       <th>内容</th>       <th>原图</th>       <th>Luban</th>       <th>Wechat</th>      </tr>     </thead>     <tbody>      <tr>       <td>截屏 720P</td>       <td>720*1280,390k</td>       <td>720*1280,87k</td>       <td>720*1280,56k</td>      </tr>      <tr>       <td>截屏 1080P</td>       <td>1080*1920,2.21M</td>       <td>1080*1920,104k</td>       <td>1080*1920,112k</td>      </tr>      <tr>       <td>拍照 13M(4:3)</td>       <td>3096*4128,3.12M</td>       <td>1548*2064,141k</td>       <td>1548*2064,147k</td>      </tr>      <tr>       <td>拍照 9.6M(16:9)</td>       <td>4128*2322,4.64M</td>       <td>1032*581,97k</td>       <td>1032*581,74k</td>      </tr>      <tr>       <td>滚动截屏</td>       <td>1080*6433,1.56M</td>       <td>1080*6433,351k</td>       <td>1080*6433,482k</td>      </tr>     </tbody>    </table>    <h2>导入</h2>    <pre>  <code class="language-java">compile 'io.reactivex:rxandroid:1.2.1'  compile 'io.reactivex:rxjava:1.1.6'    compile 'top.zibin:Luban:1.0.5'  </code></pre>    <h2>使用</h2>    <h3>Listener方式</h3>    <p>Luban内部采用io线程进行图片压缩,外部调用只需设置好结果监听即可</p>    <pre>  <code class="language-java">Luban.get(this)      .load(File)                     //传人要压缩的图片      .putGear(Luban.THIRD_GEAR)      //设定压缩档次,默认三挡      .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();    //启动压缩  </code></pre>    <h3>RxJava方式</h3>    <p>RxJava 调用方式请自行随意控制线程</p>    <pre>  <code class="language-java">Luban.get(this)          .load(file)          .putGear(Luban.THIRD_GEAR)          .asObservable()          .subscribeOn(Schedulers.io())          .observeOn(AndroidSchedulers.mainThread())          .doOnError(new Action1<Throwable>() {              @Override              public void call(Throwable throwable) {                  throwable.printStackTrace();              }          })          .onErrorResumeNext(new Func1<Throwable, Observable<? extends File>>() {              @Override              public Observable<? extends File> call(Throwable throwable) {                  return Observable.empty();              }          })          .subscribe(new Action1<File>() {              @Override              public void call(File file) {                  //TODO 压缩成功后调用,返回压缩后的图片文件              }          });</code></pre>    <p> </p>    <p> </p>