Android差量更新-1 有更新!

2017-05-03

Android差量更新-1

应用场景:省流量更新应用,只需要下载差异包,而不需要下载完整的apk进行安装。

这篇文章主要讲的是JavaEE端的实现,Android端之后的文章也会记录下来,另外使用到了Bsdiffbzip2 将源码下载下来。

我这里是在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即可使用

评论
发表评论
validate
取消