自从 Android 4.1 开始,Google引入了应用程序的增量更新。
增量更新的原理非常简单,就是将本地apk与服务器端最新版本比对,并得到差异包,用户更新App时只需要 下载差异包。例如,当前安装新浪微博V3.5,12.8 MB,新浪微博最新版V4.0,15.4MB,经过对两个版本比较之后,发现差异只有7、8M,这时候用户更新的时候只需要下载一个7、8M的差异包便可,不需要整包下载15.4M的新版微博客户端。下载差异包后,在手机端使用旧版apk+差异包,合成得到微博最新版V4.0,提醒用户安装即可。
以新浪微博客户端的升级为例,假设手机上安装的是V3.5,现在最新版是V4.0,用户需要从V3.5升级到V4.0。
弄清楚原理之后,我们就需要解决两个问题:
1.如何比对两个版本的apk生成差异包;
2.如何使用旧的apk+差异包,生成一个新apk;
(1)生成差异包
这一步需要在服务器端来实现,一般来说,apk有新版本之后,需要往后台管理端上传新apk,上传时就应该生成每一个旧版与最新版本的差异包。
假设,你的apk已经发布了3个版,1.0,2.0,3.0,这时候你要在后台发布4.0,在你上传时,就应该生成
1.0——>4.0的差异包;
2.0——>4.0的差异包;
3.0——>4.0的差异包;
选择使用这个开源二进制比较工具来实现:
http://www.daemonology.net/bsdiff/
下载后得到bsdiff-4.3.tar.gz。
其中bsdiff.c是二进制文件比对的代码;bspatch.c是二进制文件合成的代码;
我们将使用这个bsdiff来生成两个apk的patch包,并且使用bspatch.c来合成旧apk与patch包;
使用bsdiff、bspatch时,还需用到bzip2: http://www.bzip.org/downloads.html
下载后得到:bzip2-1.0.6.tar.gz。
我们需要用到bzip2-1.0.6.tar.gz中以下13个文件(这里面可能有的是不需要的,我都拷贝过来了):
将这13个文件拷贝至jni目录下,接下来,我们就调用bsdiff生成差异包,并且调用bspatch合成新包。
我是在Mac下做的,使用java开发,通过jni调用C程序(bsdiff、bzip2)。
调用,com.cundong.utils.DiffUtils.java中genDiff()方法,可以通过传入的新(newApkPath)旧(oldApkPath)apk,得到差异包(patchPath)。
1 |
public static native int genDiff(String oldApkPath, String newApkPath,String patchPath); |
调用,com.cundong.utils.PatchUtils.java中patch()方法,可以通过旧apk(oldApkPath)与差异包(patchPath),得到新apk(newApkPath)。
1 |
public static native int patch(String oldApkPath, String newApkPath,String patchPath); |
(2)使用旧apk+差异包,在客户端合成新apk
差异包已经在服务器端生成,我们只需要在客户端提示用户有更新,然后让用户来下载差异包,下载成功之后,
使用本地apk与差异包,生成新版apk。
这一步需要在Android应用中开发。
1.首先NDK编译出一个*.so,APKPatch工程负责生成 libapkpatch so,生成的so文件位于APKPatch/libs/libapkpatch.so下,其他Android工程便可以使用该so文件来合成apk。
2.调用该so文件。
任意一个Android工程,使用该so文件,拷贝至libs\armeabi中,便可以调用patch()方法,来和成旧apk+差异包。
附件中,test工程,就是一个调用该so文件的例子。
1.新包和成之后,还需要对合成升级版本的apk包及最新版本apk包进行 MD5或SHA1校验,如果校验码不一致,说明合成过程有问题,新合成的包将不能被安装。
2.增量升级成功的前提是,用户手机端必须有能够让你拷贝出来且与你服务器用于差分的版本一致的apk,这样就存在,例如,系统内置的apk无法获取到,无法进行增量升级;对于某些与你差分版本一致,但是内容有过修改的(比如 破解版apk),这样也是无法进行增量升级的,为了防止合成补丁错误,最好在补丁合成前对旧版本的apk进行校验,保证基础包的一致性。