前言

对于android项目,我们常常会遇到这样一种情况,比如某一版本的线上应用,突然发现了一个BUG,修改后发现不值当为这个BUG进行一次版本升级,那么怎么去解决在同一版本上修改线上应用的BUG呢,Google给了我们一种解决方案 Smart App update,也就是应用增量升级。

那么,何谓增量升级呢?简单的来说,也就是把同一版本的新旧两个apk进行差分,得到一个差异包,然后在旧的应用中下载该差异包和旧的apk进行合并成新的apk,然后去安装新的apk,该合成的apk和差分前新apk是一样的,这样就可以做到应用的增量升级。一般来说,获取到的差异包较之新的apk来说小了很多,这样就可以减少用户的下载流量。

增量升级需要用到jni开发,所以我们需要去生成合并差异包的so文件,这里我们需要准备的工具有:eclipse,NDK环境(这里用的是ndk9以上的版本),bsdiff文件,bzip2文件(这两个文件在下面附件中给出)。

第一步

创建一个android项目,比如SmartAppUpdateSo,设定好包名(此处为以后引用so文件的方法做准备),例如com.smartapp.update。右键点击项目 -> Android tools -> Add Native Support…,在弹出窗里填写你要生成的so文件名称,如SmartAppUpdate,这样我们就能在项目中看到jni和obj两个文件夹了,删除jni文件夹下的SmartAppUpdate.cpp文件。

第二步

新建BatchUtils工具类,在类中增加方法

private native static int patchApk(String oldApkPath, String newApkPath, String patchPath);

打开cmd界面,进入到项目的bin\classes路径下,执行javah com.smartapp.update.BatchUtils命令,我们就能在bin\classes路径下发现一个命名为com_smartapp_update_BatchUtils.h的文件,打开该文件,里面的内容为

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_smartapp_update_BatchUtils */

#ifndef _Included_com_smartapp_update_BatchUtils
#define _Included_com_smartapp_update_BatchUtils
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_smartapp_update_BatchUtils
 * Method:    patchApk
 * Signature: (Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)I
 */
JNIEXPORT jint JNICALL Java_com_smartapp_update_BatchUtils_patchApk
  (JNIEnv *, jclass, jstring, jstring, jstring);

#ifdef __cplusplus
}
#endif
#endif

第三步

将com_smartapp_update_BatchUtils.h文件拷贝到项目的jni文件夹下,然后将bzip2文件中的相关文件也拷贝到项目的jni文件夹下,需要拷贝的文件有:

blocksort.c
bzip2.c
bzip2recover.c
bzlib_private.h
bzlib.c
bzlib.h
compress.c
crctable.c
decompress.c
dlltest.c
huffman.c
mk251.c
randtable.c
spewG.c
unzcrash.c

第四步

将bsdiff中的bspatch.c文件拷贝到项目的jni文件夹中,并将其重命名为com_smartapp_update_BatchUtils.c,打开该文件,修改其中的内容:

1.将头文件替换掉,原内容为:

#if 0
__FBSDID("$FreeBSD: src/usr.bin/bsdiff/bspatch/bspatch.c,v 1.1 2005/08/06 01:59:06 cperciva Exp [        DISCUZ_CODE_3        ]quot;);
#endif

#include <bzlib.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <err.h>
#include <unistd.h>
#include <fcntl.h>

替换后的内容为:

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <err.h>
#include <unistd.h>
#include <fcntl.h>
#include <android/log.h>
#include <jni.h>

#include "bzlib.c"
#include "crctable.c"
#include "compress.c"
#include "decompress.c"
#include "randtable.c"
#include "blocksort.c"
#include "huffman.c"

#include "com_smartapp_update_BatchUtils.h"

2.将main方法名称替换成applypatch

3.在文件内添加方法

JNIEXPORT jint JNICALL Java_com_smartapp_update_BatchUtils_patchApk(JNIEnv *env,
                jobject obj, jstring old, jstring new, jstring patch) {

        char * ch[4];
        ch[0] = "bspatch";
        ch[1] = (char*) ((*env)->GetStringUTFChars(env, old, 0));
        ch[2] = (char*) ((*env)->GetStringUTFChars(env, new, 0));
        ch[3] = (char*) ((*env)->GetStringUTFChars(env, patch, 0));

        __android_log_print(ANDROID_LOG_INFO, "ApkPatchLibrary", "old = %s ", ch[1]);
        __android_log_print(ANDROID_LOG_INFO, "ApkPatchLibrary", "new = %s ", ch[2]);
        __android_log_print(ANDROID_LOG_INFO, "ApkPatchLibrary", "patch = %s ", ch[3]);

        int ret = applypatch(4, ch);

        __android_log_print(ANDROID_LOG_INFO, "ApkPatchLibrary", "applypatch result = %d ", ret);

        (*env)->ReleaseStringUTFChars(env, old, ch[1]);
        (*env)->ReleaseStringUTFChars(env, new, ch[2]);
        (*env)->ReleaseStringUTFChars(env, patch, ch[3]);

        return ret;
}

第五步

修改Android.mk文件内容

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := SmartAppUpdate
LOCAL_SRC_FILES := com_smartapp_update_BatchUtils.c

LOCAL_LDLIBS     := -lz -llog

include $(BUILD_SHARED_LIBRARY)

第六步

右键点击项目 -> properties,在弹出窗中点击Buiders,选择New…,在新弹出窗中选择Program,点击OK,此时弹出一个Edit Configuration弹窗。

1.Name中填写SmartAppUpdateBuilder

2.选择Main标签,在Location一栏中点击Browse File System,选择你NDK安装路径下的ndk-build.cmd文件,Working Directory一栏中点击Browse Workspace,选择该项目的jni文件夹

3.选择Refresh标签,勾选Refresh resources upon completion,选择The entire workspace,勾选上Recursively include sub-folders

4.选择BuildOptions,勾选Allocte Console,After a “Clean”,During manual builds,During auto builds,Specify working set of relevant resources ,点击Specify Resources,在弹出窗中勾选该项目jni文件夹,点击Apply,点击OK,这样我们就能在Buiders列表中看见新建的SmartAppUpdateBuilder,点击OK

第七步

等待片刻,我们会发现在obj文件夹下自动编译出一些文件,而其中obj\local\armeabi下的libSmartAppUpdate.so文件就是我们需要的so文件,那么生成增量升级的so文件的教程到此为止。

如何去使用该so文件进行增量升级,请看 【第二节】android增量升级之使用so文件进行增量升级

下载地址:项目文件及工具(bsdiff-4.3,bzip2-1.0.6,SmartAppUpdateSo)