NDK增量更新

658 阅读4分钟

增量更新主要是针对宿主apk更新,我们通过新旧apk生成的差分包,让用户下载差分包,然后差分包跟本地的旧包合并成为新版本的apk,提供用户安装新版本apk。

增量更新的作用:通过新旧包生成差分包,差分包提供给用户下载,由于差分包比较小,可以为用户节省流量。同时,由于差分包比较小,也可以在后台悄悄下载差分包,然后跟旧包合成新包,提供给用户安装。这种方式大大的节省了用户的流量。

本次增量更新主要通过开源库BsDiff,来实现拆分和合并的功能。首先需要下载BsDiff的源码。由于BsDiff依赖于Bzip2,所以需要把Bzip2的源码也要下载下来。

下载包.png

准备好上述文件就可以做增量开发了。

差分包

增量更新,首先得到差分包,而差分包这一部分往往是后台开发人员需要做的事情,客户端只需要提供最新包给后台,后台人员需要将最新包和旧包生成差分包。不过也需要熟悉jni开发才能做这样的事情。

新建一个java项目,编写生成差分包的本地方法,并且使用javah生成.h头文件

运行结果.png

Visual Studio新建一个项目JNITest

JNITest.png

将下载的bsdiff源码包解压,将头文件和源文件复制到JNITest项目下

运行结果.png

JNITest项目文件如下

运行结果.png

JNITest项目,头文件添加现有项,将项目目录下的头文件添加进来。源文件添加现有项,将项目下的源文件添加进来。

添加现有项.png

启动调试该项目,出现错误,提示:用了不安全的函数 以及 过时的函数。需要添加宏定义。打开项目属性,在命令行下添加宏定义。

添加宏定义.png

再次运行,还是出现错误,由于VS对函数的严谨性,有些函数编译不通过,所以去掉SDL检查。

SDL检查.png

再次编译就通过了。

打开bsdiff.cpp源文件,添加#include "cn_com_diff_NativeDiff.h" 和 添加该头文件的方法。

运行结果.png

运行结果.png

修改bsdiff.cpp的入口函数的名字,改成通过jni调用本地方法,本地方法在调用入口函数。 通过解读bsdiff.cpp入口函数,可知其第一个参数一定要=4,第二参数也就是数组:数组的第一个元素是无效的,第2、3、4个元素分别是:oldfile、newfile、 patchfile。也就是旧文件,新文件,差分包的路径。

源文件代码.png

修改入口函数名.png

在源文件实现本地方法,并且调用刚刚修改的入口函数方法

运行结果.png

修改配置,生成动态库

修改配置.png

生成解决方案.png

调用本地方法,生成差分包:

调用.png

得到差分包.png

下面是给出的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)编写本地方法

本地方法.png

3)通过javah 得到头文件

运行结果.png

4)创建jni文件夹,把头文件、bspatch.cpp源文件 和 bzip2文件放到该jni目录下。

jni目录.png

5)跟差分包的实现一样,引入头文件bzip的源文件修改入口函数名,实现头文件的本地方法,在本地方法中调用修改后的入口函数

引入头文件.png

修改入口方法.png

本地方法实现.png

调用:

运行结果.png

下面给出各位文件的源码

目录.png

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;
	
}

合并完成之后直接调用安装就好了。

运行结果.png

以上就是使用bsdiff实现增量更新的流程。