最近项目上需要做APK的差分升级,网上查了些前人的技术文档学习来下,这里也总结下自己的经验。
差分升级主要流程就是,拿新APK和旧apk进行比较生成patch包,放在服务器上让用户下载,用户下载完成后本地apk程序再将本地的apk和patch包整合成一个最新的apk让用户进行安装。
我们需要如下2套源码来实现以上功能 bsdiff/bzlib2
这里我还要介绍一下其实android已经集成了以上两个模块,我们可以在源码的external目录里面看到bsdiff和bzip2两个文件夹对应的就是上面的2套源码。
我们在能编译的源码环境中mmm external/bsdiff看如下截图
这里out/host/linux-x86/obj32/EXECUTABLES/bspatch_intermediates/bsdiff就是用来生成patch的,out/host/linux-x86/obj32/EXECUTABLES/bspatch_intermediates/bspatch是用来整合patch生成新apk的。
所以我们可以拿bsdiff去生成patch包,然后把bspatch相关代码整合成so库加载到我们应用内去运行再生成最新的整包这整个差分升级功能就完成了。
bsdiff命令如下,大家可以自己尝试oldfile是旧包路径+文件名,newfile是新包的,patchfile是生成的patch我们这里可以随便命名为XXX.patch
./bsdiff oldfile newfile patchfile
以上前期准备完成了,现在直奔主题贴上android上功能实现需要的代码
1、新建jni目录存放so库所需源码我只写了3个文件
bs_patch.c
bs_patch.h
Android.mk
bs_patch.h
#ifndef BS_PATCH_H
#define BS_PATCH_H
JNIEXPORT jint JNICALL Java_com_android_updatetest_util_PatchUtil_applyPatchToApk
(JNIEnv *, jclass, jstring, jstring, jstring);
#endif
bs_patch.c这个文件功能参考源码external/bsdiff/bspatch.c加了jni接口
#include <jni.h>
#include "bs_patch.h"
#include <bzlib.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <err.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/types.h>
static off_t offtin(u_char *buf) {
off_t y;
y = buf[7] & 0x7F;
y = y * 256;
y += buf[6];
y = y * 256;
y += buf[5];
y = y * 256;
y += buf[4];
y = y * 256;
y += buf[3];
y = y * 256;
y += buf[2];
y = y * 256;
y += buf[1];
y = y * 256;
y += buf[0];
if (buf[7] & 0x80)
y = -y;
return y;
}
int applypatch(int argc, const char* argv[]) {
FILE * f, *cpf, *dpf, *epf;
BZFILE * cpfbz2, *dpfbz2, *epfbz2;
int cbz2err, dbz2err, ebz2err;
int fd;
ssize_t oldsize, newsize;
ssize_t bzctrllen, bzdatalen;