Android差量更新-1
應(yīng)用場(chǎng)景:省流量更新應(yīng)用璧微,只需要下載差異包艾帐,而不需要下載完整的apk進(jìn)行安裝誓沸。
這篇文章主要講的是JavaEE端的實(shí)現(xiàn),Android端之后的文章也會(huì)記錄下來(lái)茸炒,另外使用到了Bsdiff 與 bzip2 將源碼下載下來(lái)。
我這里是在Linux下編譯源碼阵苇,Windows上嘗試了挺長(zhǎng)時(shí)間 編譯不起來(lái)壁公,應(yīng)該還是我太菜了貼一張Windwos編譯圖 缺的東西挺多 后來(lái)直接放棄了,直接Linux進(jìn)行編譯慎玖。
解壓從Bsdiff上下載來(lái)的源碼贮尖,如圖:
打開(kāi)我們的Java工程。創(chuàng)建一個(gè)Diff工具類(lèi) 也是一個(gè)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);
}
打開(kāi)終端趁怔,切到DiffUtils所在的目錄湿硝,執(zhí)行Javac編譯 會(huì)在目錄下生成一個(gè)class文件
javac DiffUtils.java
接下來(lái) 切換到代碼最上層目錄 即src目錄下 執(zhí)行javah命令 生成.h頭文件。
javah -classpath . -jni io.javac.diff_javaee.Utils.DiffUtils
這時(shí)候工程的src目錄下應(yīng)該生成了一個(gè)io_javac_diff_javaee_Utils_DiffUtils.h文件
然后把解壓bsdiff后的文件夾當(dāng)中润努,找到bsdiff.c這個(gè)文件关斜,將其復(fù)制到src目錄,與剛剛生成的.h文件同級(jí)铺浇。
使用編輯器痢畜,打開(kāi)bsdiff.c文件,加入jni.h 與剛剛生成的io_javac_diff_javaee_Utils_DiffUtils.h的引用
打開(kāi)io_javac_diff_javaee_Utils_DiffUtils.h文件鳍侣,其中可以看到有一個(gè)方法丁稀。
然后我們需要打開(kāi)bsdiff.c文件,在當(dāng)中實(shí)現(xiàn)這個(gè)方法倚聚。
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的圖 我這里對(duì)main方法更名了diffFile 不然編譯不會(huì)通過(guò)的线衫。
整理這些C源碼。然后Java工程src中的C源碼可以刪除了惑折。
找個(gè)地方新建一個(gè)文件夾授账,將bsdiff.c io_javac_diff_javaee_Utils_DiffUtils.h文件都復(fù)制進(jìn)去
除此之外枯跑,解壓下好的bzip2 找到以下文件,并且復(fù)制到文件夾當(dāng)中去:
blocksort.c bzip2.c bzlib.c bzlib.h bzlib_private.h compress.c crctable.c decompress.c huffman.c randtable.c
目錄 列表
終端切到目錄下 執(zhí)行編譯命令:
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/ 這個(gè)是jdk安裝目錄include目錄下的linux文件夾路徑
-I/usr/local/jdk1.8.0_121/include/ 這個(gè)也在jdk安裝目錄
-I/home/so 這個(gè)是整理好的C源碼的文件夾路徑
libdiff.so 要生成的so文件名 一定要以lib開(kāi)頭
執(zhí)行成功會(huì)在目錄下生成libdiff.so文件白热。
生成差異包
先打印一下依賴(lài)庫(kù)的路徑: System.out.println("java.library.path:" + System.getProperty("java.library.path"));
接著會(huì)輸出所有已添加的依賴(lài)庫(kù)的路徑敛助,隨便找一個(gè)路徑,把so文件丟進(jìn)去即可屋确,我這里的話 把so文件丟到了/usr/lib64/目錄
public class MainDiff {
static {
System.loadLibrary("diff");//裝載動(dòng)態(tài)鏈接庫(kù)纳击,記得把開(kāi)頭的“l(fā)ib”去掉+
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中執(zhí)行
生成后的文件
SO成品 可以跳過(guò)編譯這些步驟,直接配合DiffUtils即可使用