上一节我们学习了NDK来处理文件的拆分和合并操作,那时候我们纯手工来敲C语言的代码,今天我们来用C语言代码搞搞NDK的增量更新差分包,
那什么是增量更新呢?再说这个之前我们有必要说一下热修复,什么又是热修复呢?
对于热修复就是打补丁,当有重大的bug的时候需要使用热修复的功能,这样避免了重新打包并发布所带来的成本高、效率低,热修复会使用到PathClassLoader和DexClassLoader类。
增量更新顾名思义就是用来apk的版本更新,增量则通过生成两个apk的增量文件patch,然后将旧版本的apk进行合并达到更新的目的。
做增量更新需要注意:
- 新版本和旧版本的apk签名要一致
- 新版本的大小要增加
- 最好要生成几个版本的增量文件,每个版本去下载对应的文件去做增量更新
正文
我们从两方面来进行讲解:根据旧版本和新版本的apk生成增量文件、根据增量文件客户端做相应的增量更新
生成增量更新
生成增量更新的操作都是放到后台来进行,首先我们先去配置好Tomcat和NDK环境
接下来要生成增量文件就需要使用第三方的bsdiff库
http://www.daemonology.net/bsdiff/
在这里面不仅有差分文件还有合并的文件,因为bsdiff和bspatch都使用到了bzip2,所以我们还需要去下载bzip2库
http://www.bzip.org/
这样我们就可以正式开发服务端生成增量更新的功能了。
-
导入bsdiff.c和bzip2里的文件
在编译的时候会遇到一些坑,这里列举一下:
- 会提示_CRT_SECURE_NO_WARNINGS _CRT_NONSTDC_NO_DEPRECATE没有使用,这里因为涉及文件比较多,我们在命令行中进行添加
- 不能将参数 1 从 CHAR [1024] 转换为 LPWSTR,解决:在项目属性->常规中,把Uicode改成多字符段
- 编译的时候会报字段过时,解决关闭dsl检查
-
在bsdiff.cpp里面有main方法,那怎么来加入到NDK里面来呢?这时会想到将main改成普通方法,使用native方法调用即可
JNIEXPORT void JNICALL Java_com_lypop_BsDiff_diff(JNIEnv *env, jclass jcls, jstring oldfile_jstr, jstring newfile_jstr, jstring patchfile_jstr){ int argc = 4; char* oldfile = (char*)(env)->GetStringUTFChars( oldfile_jstr, NULL); char* newfile = (char*)(env)->GetStringUTFChars(newfile_jstr, NULL); char* patchfile = (char*)(env)->GetStringUTFChars(patchfile_jstr, NULL); char *argv[4]; argv[0] = "bspatch"; argv[1] = oldfile; argv[2] = newfile; argv[3] = patchfile; bsdiff_main(argc,argv); //释放资源 env->ReleaseStringUTFChars(oldfile_jstr, oldfile); env->ReleaseStringUTFChars(newfile_jstr, newfile); env->ReleaseStringUTFChars(patchfile_jstr, patchfile); };
在写这个方法之前需要先看一下main方法
int bsdiff_main(int argc,char *argv[])
{
if(argc!=4) errx(1,"usage: %s oldfile newfile patchfile\n",argv[0]);
......
}
由源码可以看出第一个参数可有可无,重点是最后的那三个参数,分别代表的是oldfile、newfile、patchfile,然后根据要传入的参数写相应的native方法并生成.h文件,最后生成相应的dll动态库让后台进行调用生成相应的增量patch文件
进行增量更新
在Android这边我们开启了一个异步任务用来请求服务器的patch文件,将patch文件合并到apk中进行增量更新
class ApkUpdateTask extends AsyncTask<Void, Void, Boolean>{
@Override
protected Boolean doInBackground(Void... params) {
try {
//1.下载差分包
File patchFile = DownloadUtils.download(Constants.URL_PATCH_DOWNLOAD);
//获取当前应用的apk文件
String oldfile = ApkUtils.getSourceApkPath(MainActivity.this, getPackageName());
//2.合并得到最新版本的APK文件
String newfile = Constants.NEW_APK_PATH;
String patchfile = patchFile.getAbsolutePath();
BsPatch.patch(oldfile, newfile, patchfile);
合并的native方法
/**
* 合并
* @param oldfile
* @param newfile
* @param patchfile
*/
public native static void patch(String oldfile,String newfile,String patchfile);
static{
System.loadLibrary("bspatch");
}