Android差量更新-1
应用场景:省流量更新应用,只需要下载差异包,而不需要下载完整的apk进行安装。
这篇文章主要讲的是JavaEE端的实现,Android端之后的文章也会记录下来,另外使用到了Bsdiff 与 bzip2 将源码下载下来。
我这里是在Linux下编译源码,Windows上尝试了挺长时间 编译不起来,应该还是我太菜了贴一张Windwos编译图 缺的东西挺多 后来直接放弃了,直接Linux进行编译。
解压从Bsdiff上下载来的源码,如图:
打开我们的Java工程。创建一个Diff工具类 也是一个JNI接口
package io.javac.diff_javaee.Utils;
/**
* Created by Pencilso on 2017/5/2.
*/
public class DiffUtils {
public static native int diffFile(String oldPath, String newPath , String patchPath);
}
打开终端,切到DiffUtils所在的目录,执行Javac编译 会在目录下生成一个class文件
javac DiffUtils.java
接下来 切换到代码最上层目录 即src目录下 执行javah命令 生成.h头文件。
javah -classpath . -jni io.javac.diff_javaee.Utils.DiffUtils
这时候工程的src目录下应该生成了一个io_javac_diff_javaee_Utils_DiffUtils.h文件
然后把解压bsdiff后的文件夹当中,找到bsdiff.c这个文件,将其复制到src目录,与刚刚生成的.h文件同级。
使用编辑器,打开bsdiff.c文件,加入jni.h 与刚刚生成的io_javac_diff_javaee_Utils_DiffUtils.h的引用
打开io_javac_diff_javaee_Utils_DiffUtils.h文件,其中可以看到有一个方法。
然后我们需要打开bsdiff.c文件,在当中实现这个方法。
JNIEXPORT jint JNICALL Java_io_javac_diff_1javaee_Utils_DiffUtils_diffFile
(JNIEnv *env, jclass cls, jstring old, jstring new, jstring patch){
int argc=4;
char * argv[argc];
argv[0]="bsdiff";
argv[1]=(char*)((*env)->GetStringUTFChars(env,old, 0));
argv[2]=(char*)((*env)->GetStringUTFChars(env,new, 0));
argv[3]=(char*)((*env)->GetStringUTFChars(env,patch, 0));
int ret=diffFile(argc, argv);
(*env)->ReleaseStringUTFChars(env,old,argv[1]);
(*env)->ReleaseStringUTFChars(env,new,argv[2]);
(*env)->ReleaseStringUTFChars(env,patch,argv[3]);
return ret;
}
贴张bsdiff.c的图 我这里对main方法更名了diffFile 不然编译不会通过的。
整理这些C源码。然后Java工程src中的C源码可以删除了。
找个地方新建一个文件夹,将bsdiff.c io_javac_diff_javaee_Utils_DiffUtils.h文件都复制进去
除此之外,解压下好的bzip2 找到以下文件,并且复制到文件夹当中去:
blocksort.c bzip2.c bzlib.c bzlib.h bzlib_private.h compress.c crctable.c decompress.c huffman.c randtable.c
目录 列表
终端切到目录下 执行编译命令:
gcc -I/usr/local/jdk1.8.0_121/include/linux/ -I/usr/local/jdk1.8.0_121/include/ -I/home/so -fPIC -shared -o libdiff.so bsdiff.c bzlib.c bzip2.c blocksort.c compress.c crctable.c decompress.c huffman.c randtable.c
-I/usr/local/jdk1.8.0_121/include/linux/ 这个是jdk安装目录include目录下的linux文件夹路径
-I/usr/local/jdk1.8.0_121/include/ 这个也在jdk安装目录
-I/home/so 这个是整理好的C源码的文件夹路径
libdiff.so 要生成的so文件名 一定要以lib开头
执行成功会在目录下生成libdiff.so文件。
生成差异包
先打印一下依赖库的路径: System.out.println("java.library.path:" + System.getProperty("java.library.path"));
接着会输出所有已添加的依赖库的路径,随便找一个路径,把so文件丢进去即可,我这里的话 把so文件丢到了/usr/lib64/目录
public class MainDiff {
static {
System.loadLibrary("diff");//装载动态链接库,记得把开头的“lib”去掉+
System.out.println("java.library.path:" + System.getProperty("java.library.path"));
}
public static void main(String args[]) {
System.out.println("正在生成差异包");
String oldApk = "/Diff-JavaEE/apk/old.apk"; //旧的apk包路径
String newApk = "/Diff-JavaEE/apk/new.apk"; //新的apk包路径
String patch = "/Diff-JavaEE/apk/new.patchPath";//差异包生成路径
DiffUtils.diffFile(oldApk, newApk, patch);
System.out.println("差异包生成成功");
}
}
Linux中执行
生成后的文件
SO成品 可以跳过编译这些步骤,直接配合DiffUtils即可使用
网友评论