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 即可使用
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于