增量更新主要是针对宿主apk更新,我们通过新旧apk生成的差分包,让用户下载差分包,然后差分包跟本地的旧包合并成为新版本的apk,提供用户安装新版本apk。
增量更新的作用:通过新旧包生成差分包,差分包提供给用户下载,由于差分包比较小,可以为用户节省流量。同时,由于差分包比较小,也可以在后台悄悄下载差分包,然后跟旧包合成新包,提供给用户安装。这种方式大大的节省了用户的流量。
本次增量更新主要通过开源库BsDiff,来实现拆分和合并的功能。首先需要下载BsDiff的源码。由于BsDiff依赖于Bzip2,所以需要把Bzip2的源码也要下载下来。
准备好上述文件就可以做增量开发了。
差分包
增量更新,首先得到差分包,而差分包这一部分往往是后台开发人员需要做的事情,客户端只需要提供最新包给后台,后台人员需要将最新包和旧包生成差分包。不过也需要熟悉jni开发才能做这样的事情。
新建一个java项目,编写生成差分包的本地方法,并且使用javah生成.h头文件
在Visual Studio新建一个项目JNITest
将下载的bsdiff源码包解压,将头文件和源文件复制到JNITest项目下
JNITest项目文件如下
JNITest项目,头文件添加现有项,将项目目录下的头文件添加进来。源文件添加现有项,将项目下的源文件添加进来。
启动调试该项目,出现错误,提示:用了不安全的函数 以及 过时的函数。需要添加宏定义。打开项目属性,在命令行下添加宏定义。
再次运行,还是出现错误,由于VS对函数的严谨性,有些函数编译不通过,所以去掉SDL检查。
再次编译就通过了。
打开bsdiff.cpp源文件,添加#include "cn_com_diff_NativeDiff.h" 和 添加该头文件的方法。
修改bsdiff.cpp的入口函数的名字,改成通过jni调用本地方法,本地方法在调用入口函数。
通过解读bsdiff.cpp入口函数,可知其第一个参数一定要=4,第二参数也就是数组:数组的第一个元素是无效的,第2、3、4个元素分别是:oldfile、newfile、 patchfile。也就是旧文件,新文件,差分包的路径。
在源文件实现本地方法,并且调用刚刚修改的入口函数方法。
修改配置,生成动态库
调用本地方法,生成差分包:
下面是给出的java文件源码:
NativeDiff.java
public class NativeDiff {
public native static void diff(String oldfile, String newfile, String patchfile);
static{
System.loadLibrary("JNITest");
}
}
Diff.java
public class Diff {
public static void main(String[] args) {
// TODO Auto-generated method stub
System.out.println("start");
NativeDiff.diff(ConstantsWin.OLD_APK_PATH, ConstantsWin.NEW_APK_PATH, ConstantsWin.PATCH_PATH);
System.out.println("end");
}
}
ConstantsWin.java
public class ConstantsWin {
public static final String OLD_APK_PATH = "D:/app/old_app.apk";
public static final String NEW_APK_PATH = "D:/app/old_app.apk";
public static final String PATCH_PATH = "D:/app/apk.patch";
}
上面就是通过bsdiff开源库,得到差分包的过程。
NDK实现合并
合并的正常流程是:从后台下载差分包,得到差分包,然后获取本地旧包的路径,通过NDK合并成新版本的apk提供给用户安装。
1)配置NDK环境。 2)编写本地方法
3)通过javah 得到头文件
4)创建jni文件夹,把头文件、bspatch.cpp源文件 和 bzip2文件放到该jni目录下。
5)跟差分包的实现一样,引入头文件和bzip的源文件,修改入口函数名,实现头文件的本地方法,在本地方法中调用修改后的入口函数。
调用:
下面给出各位文件的源码
MainActivity
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
this.findViewById(R.id.button).setOnClickListener(mOnClickListener);
this.findViewById(R.id.button01).setOnClickListener(mOnClickListener);
this.findViewById(R.id.button02).setOnClickListener(mOnClickListener);
}
private View.OnClickListener mOnClickListener = new View.OnClickListener() {
@Override
public void onClick(View view) {
switch (view.getId()) {
case R.id.button:
SubContainer.getInstance().gowelcome(new SubBundle(MainActivity.this));
break;
case R.id.button01: {
try {
//当前apk的路径
String oldfile = ApkUtils.getSourceApkPath(MainActivity.this, getPackageName());
//2.合并得到最新版本的APK文件
String newfile = Constants.NEW_APK_PATH;
//差分包路径
String patchfile = Constants.PATCH_FILE_PATH;
BsPatch.patch(oldfile, newfile, patchfile);
} catch (Exception e) {
e.printStackTrace();
}
}
break;
case R.id.button02:
ApkUtils.installApk(MainActivity.this, Constants.NEW_APK_PATH);
break;
default:
break;
}
}
};
}
BsPatch
public class BsPatch {
public native static void patch(String oldfile, String newfile, String patchfile);
static{
System.loadLibrary("bspatch");
}
}
ApkUtils
public class ApkUtils {
public static boolean isInstalled(Context context, String packageName) {
PackageManager pm = context.getPackageManager();
boolean installed = false;
try {
pm.getPackageInfo(packageName, PackageManager.GET_ACTIVITIES);
installed = true;
} catch (Exception e) {
e.printStackTrace();
}
return installed;
}
/**
* 获取已安装Apk文件的源Apk文件
* 如:/data/app/my.apk
*
* @param context
* @param packageName
* @return
*/
public static String getSourceApkPath(Context context, String packageName) {
if (TextUtils.isEmpty(packageName))
return null;
try {
ApplicationInfo appInfo = context.getPackageManager()
.getApplicationInfo(packageName, 0);
return appInfo.sourceDir;
} catch (NameNotFoundException e) {
e.printStackTrace();
}
return null;
}
/**
* 安装Apk
*
* @param context
* @param apkPath
*/
public static void installApk(Context context, String apkPath) {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setDataAndType(Uri.parse("file://" + apkPath),
"application/vnd.android.package-archive");
context.startActivity(intent);
}
}
Constants
public class Constants {
public static final String PATCH_FILE = "apk.patch";
public static final String SD_CARD = Environment.getExternalStorageDirectory() + File.separator + "/myapp/";
//新版本apk的目录
public static final String NEW_APK_PATH = SD_CARD+"new_app.apk";
public static final String PATCH_FILE_PATH = SD_CARD+PATCH_FILE;
}
合并完成之后直接调用安装就好了。
以上就是使用bsdiff实现增量更新的流程。