Android热修复技术之阿里Sophix使用详解

前言

最近刚刚学习了一波热更新技术,之前也打算去研究的,当时看了鸿洋大神博客里对热修复Tinker的讲解,看的不是很懂,看完后也没有动手去实现,就不了了之。这两天又有兴趣研究了一下,对比了当前一些比较流行的热更新技术之后,发现市场上口碑比较好的有阿里巴巴旗下的Sophix热更新方案和腾讯的 Tinker热更新方案,然后自己实现了一遍Sophix方案,个人感觉比较容易实现。下面来介绍一下具体操作。

Sophix --阿里终极热修复方案

下面就来介绍项目集成下看看具体的使用效果:

快速集成

1.1Android SDK及工具下载

阿里云Sophix 3.0版本现已上线!

Sophix提供了一套更加完美的客户端服务端一体的热更新方案,做到了图形界面一键打包、加密传输、签名校验和服务端控制发布与灰度功能,让你用最少的时间实现最强大可靠的全方位热更新。

一张表格来说明一下各个版本热修复的差别:

image

说明:

[0] 部分情况指的是构造方法、参数数目大于8或者参数包括long,double,float基本类型的方法。

[1] 冷启动方式,指的是需要重启app在下次启动时才能生效。

[2] 对于Andfix及Hotfix 1.X能够支持的代码变动情况,都能做到即时生效。而对于其他代码变动较大的情况,会走冷启动方式,此时就无法做到即时生效。

[3] Hotfix 1.X已经支持绝大部分主流手机,只是在X86设备以及修改了虚拟机底层结构的ROM上不支持。

[4] 由于支持了资源和库,如果有这些方面的更新,就会导致的补丁变大一些,这个是很正常的。并且由于只包含差异的部分,所以补丁已经是最大程度的小了。

[5] 提供服务端的补丁发布和停发、版本控制和灰度功能,存储开发者上传的补丁包。

1.2 集成准备

1.2.1 android studio集成方式

gradle远程仓库依赖, 打开项目找到app的build.gradle文件,添加如下配置:

添加maven仓库地址:

repositories {

  maven {

  url"http://maven.aliyun.com/nexus/content/repositories/relea  ses"

 }

}

添加gradle坐标版本依赖:

compile'com.aliyun.ams:alicloud-android-hotfix:3.2.0'

注意,若SDK集成过程中出现UTDID冲突,请参考:阿l里云-移动云产品SDK UTDID冲突解决方案

如若仓库访问失败, 那么用本地依赖的方式进行依赖, SDK下载见“1.5 客户端本地SDK及DEMO下载”节。

1.2.2 eclipse集成方式

下载OneSDk.zip,解压出hotfix_core-release.aar文件后再解压这个aar文件

复制解压文件jni目录下的libsophix.so到自己的jni目录下, eclipse jni目录一般指的就是项目libs目录

复制utdid4all-1.1.5.3_proguard.jar和alicloud-android-utils-1.0.3.jar文件到项目libs目录下

重命名classes.jar为sophix.jar并复制到项目libs目录下

合并AndroidManifest.xml文件中的内容到本项目AndroidManifest.xml文件

编译期间报utdid类重复异常, 那么步骤2中添加的utdid4all-1.1.5.3_proguard.jar从项目libs目录移除即可

1.2.3 权限说明

Sophix SDK使用到以下权限

READ_EXTERNAL_STORAGE权限属于Dangerous Permissions,仅调试工具获取外部补丁需要,不影响线上发布的补丁加载,调试时请自行做好android6.0以上的运行时权限获取。

1.2.4 配置AndroidManifest文件

在AndroidManifest.xml中间的application节点下添加如下配置:

android:name="com.taobao.android.hotfix.IDSECRET"

android:value="App ID" />

android:name="com.taobao.android.hotfix.APPSECRET"

android:value="App Secret" />

android:name="com.taobao.android.hotfix.RSASECRET"

android:value="RSA密钥" />

将上述value中的值分别改为通过平台HotFix服务申请得到的App Secret和RSA密钥,出于安全考虑,建议使用setSecretMetaData这个方法进行设置,详见1.3.2.1的方法说明。

注:App ID/App Secret将被用于计量计费,请妥善保管注意安全。

1.2.5 混淆配置

#基线包使用,生成mapping.txt

-printmappingmapping.txt

#生成的mapping.txt在app/buidl/outputs/mapping/release路径下,移动到/app路径下

#修复后的项目使用,保证混淆结果一致

-applymappingmapping.txt

#hotfix

-keep classcom.taobao.sophix.*{;}

-keep classcom.ta.utdid2.device.*{;}

#防止inline

-dontoptimize

1.3 SDK接口使用说明

1.3.1 接入范例

initialize的调用应该尽可能的早,必须在Application.attachBaseContext()或者Application.onCreate()的最开始进行SDK初始化操作,初始化之前不能用到其他自定义类,否则极有可能导致崩溃。而查询服务器是否有可用补丁的操作可以在后面的任意地方。

// initialize最好放在attachBaseContext最前面,初始化直接在Application类里面,切勿封装到其他类

SophixManager.getInstance().setContext(this)

            .setAppVersion(appVersion)

            .setAesKey(null)

            .setEnableDebug(true)

            .setPatchLoadStatusStub(new PatchLoadStatusListener() {

                @Override

                public voidonLoad(final int mode, final int code, final String info, final int handlePatchVersion) {

                    // 补丁加载回调通知

                    if (code== PatchStatus.CODE_LOAD_SUCCESS) {

                        // 表明补丁加载成功

                    } else if (code== PatchStatus.CODE_LOAD_RELAUNCH) {

                        // 表明新补丁生效需要重启. 开发者可提示用户或者强制重启;

                        // 建议: 用户可以监听进入后台事件, 然后调用killProcessSafely自杀,以此加快应用补丁,详见1.3.2.3

                    } else {

                        // 其它错误信息, 查看PatchStatus类说明

                    }

                }

            }).initialize();

// queryAndLoadNewPatch不可放在attachBaseContext 中,否则无网络权限,建议放在后面任意时刻,如onCreate中

SophixManager.getInstance().queryAndLoadNewPatch();

1.3.2 接口说明

1.3.2.1 initialize方法

initialize(): <必选>

该方法主要做些必要的初始化工作以及如果本地有补丁的话会加载补丁, 但不会自动请求补丁。因此需要自行调用queryAndLoadNewPatch方法拉取补丁。这个方法调用需要尽可能的早, 推荐在Application的onCreate方法中调用, initialize()方法调用之前你需要先调用如下几个方法, 方法调用说明如下:

setContext(application): <必选> 传入入口Application即可

setAppVersion(appVersion): <必选> 应用的版本号

setSecretMetaData(idSecret, appSecret, rsaSecret): <可选,推荐使用> 三个Secret分别对应AndroidManifest里面的三个,可以不在AndroidManifest设置而是用此函数来设置Secret。放到代码里面进行设置可以自定义混淆代码,更加安全,此函数的设置会覆盖AndroidManifest里面的设置,如果对应的值设为null,默认会在使用AndroidManifest里面的。

setEnableDebug(isEnabled): <可选> isEnabled默认为false, 是否调试模式, 调试模式下会输出日志以及不进行补丁签名校验. 线下调试此参数可以设置为true, 查看日志过滤TAG:Sophix, 同时强制不对补丁进行签名校验, 所有就算补丁未签名或者签名失败也发现可以加载成功. 但是正式发布该参数必须为false, false会对补丁做签名校验, 否则就可能存在安全漏洞风险

setAesKey(aesKey): <可选> 用户自定义aes秘钥, 会对补丁包采用对称加密。这个参数值必须是16位数字或字母的组合,是和补丁工具设置里面AES Key保持完全一致, 补丁才能正确被解密进而加载。此时平台无感知这个秘钥, 所以不用担心阿里云移动平台会利用你们的补丁做一些非法的事情。

setPatchLoadStatusStub(new PatchLoadStatusListener()): <可选> 设置patch加载状态监听器, 该方法参数需要实现PatchLoadStatusListener接口, 接口说明见1.3.2.2说明

setUnsupportedModel(modelName, sdkVersionInt):<可选> 把不支持的设备加入黑名单,加入后不会进行热修复。modelName为该机型上Build.MODEL的值,这个值也可以通过adb shell getprop | grep ro.product.model取得。sdkVersionInt就是该机型的Android版本,也就是Build.VERSION.SDK_INT,若设为0,则对应该机型所有安卓版本。

1.3.2.2 queryAndLoadNewPatch方法

该方法主要用于查询服务器是否有新的可用补丁. SDK内部限制连续两次queryAndLoadNewPatch()方法调用不能短于3s, 否则的话就会报code:19的错误码. 如果查询到可用的话, 首先下载补丁到本地, 然后

应用原本没有补丁, 那么如果当前应用的补丁是热补丁, 那么会立刻加载(不管是冷补丁还是热补丁). 如果当前应用的补丁是冷补丁, 那么需要重启生效.

应用已经存在一个补丁, 请求发现有新补丁后,本次不受影响。并且在下次启动时补丁文件删除, 下载并预加载新补丁。在下下次启动时应用新补丁。

补丁在后台发布之后, 并不会主动下行推送到客户端, 需要手动调用queryAndLoadNewPatch方法查询后台补丁是否可用.

只会下载补丁版本号比当前应用存在的补丁版本号高的补丁, 比如当前应用已经下载了补丁版本号为5的补丁, 那么只有后台发布的补丁版本号>5才会重新下载.

同时1.4.0以上版本服务后台上线了“一键清除”补丁的功能, 所以如果后台点击了“一键清除”那么这个方法将会返回code:18的状态码. 此时本地补丁将会被强制清除, 同时不清除本地补丁版本号

1.3.2.3 killProcessSafely方法

可以在PatchLoadStatusListener监听到CODE_LOAD_RELAUNCH后在合适的时机,调用此方法杀死进程。注意,不可以直接Process.killProcess(Process.myPid())来杀进程,这样会扰乱Sophix的内部状态。因此如果需要杀死进程,建议使用这个方法,它在内部做一些适当处理后才杀死本进程。

1.3.2.4 cleanPatches()方法]

清空本地补丁,并且不再拉取被清空的版本的补丁。正常情况下不需要开发者自己调用,因为Sophix内部会判断对补丁引发崩溃的情况进行自动清空。

1.3.2.5 PatchLoadStatusListener接口

该接口需要自行实现并传入initialize方法中, 补丁加载状态会回调给该接口, 参数说明如下:

mode: 无实际意义, 为了兼容老版本, 默认始终为0

code: 补丁加载状态码, 详情查看PatchStatus类说明

info: 补丁加载详细说明

handlePatchVersion: 当前处理的补丁版本号, 0:无 -1:本地补丁 其它:后台补丁

常见状态码说明如下: 一个补丁的加载一般分为三个阶段: 查询/预加载/加载

//兼容老版本的code说明

int CODE_LOAD_SUCCESS = 1;//加载阶段, 成功

int CODE_ERR_INBLACKLIST = 4;//加载阶段, 失败设备不支持

int CODE_REQ_NOUPDATE = 6;//查询阶段, 没有发布新补丁

int CODE_REQ_NOTNEWEST = 7;//查询阶段, 补丁不是最新的

int CODE_DOWNLOAD_SUCCESS = 9;//查询阶段, 补丁下载成功

int CODE_DOWNLOAD_BROKEN = 10;//查询阶段, 补丁文件损坏下载失败

int CODE_UNZIP_FAIL = 11;//查询阶段, 补丁解密失败

int CODE_LOAD_RELAUNCH = 12;//预加载阶段, 需要重启

int CODE_REQ_APPIDERR = 15;//查询阶段, appid异常

int CODE_REQ_SIGNERR = 16;//查询阶段, 签名异常

int CODE_REQ_UNAVAIABLE = 17;//查询阶段, 系统无效

int CODE_REQ_SYSTEMERR = 22;//查询阶段, 系统异常

int CODE_REQ_CLEARPATCH = 18;//查询阶段, 一键清除补丁

int CODE_PATCH_INVAILD = 20;//加载阶段, 补丁格式非法

//查询阶段的code说明

int CODE_QUERY_UNDEFINED = 31;//未定义异常

int CODE_QUERY_CONNECT = 32;//连接异常

int CODE_QUERY_STREAM = 33;//流异常

int CODE_QUERY_EMPTY = 34;//请求空异常

int CODE_QUERY_BROKEN = 35;//请求完整性校验失败异常

int CODE_QUERY_PARSE = 36;//请求解析异常

int CODE_QUERY_LACK = 37;//请求缺少必要参数异常

//预加载阶段的code说明

int CODE_PRELOAD_SUCCESS = 100;//预加载成功

int CODE_PRELOAD_UNDEFINED = 101;//未定义异常

int CODE_PRELOAD_HANDLE_DEX = 102;//dex加载异常

int CODE_PRELOAD_NOT_ZIP_FORMAT = 103;//基线dex非zip格式异常

int CODE_PRELOAD_REMOVE_BASEDEX = 105;//基线dex处理异常

//加载阶段的code说明 分三部分dex加载, resource加载, lib加载

//dex加载

int CODE_LOAD_UNDEFINED = 71;//未定义异常

int CODE_LOAD_AES_DECRYPT = 72;//aes对称解密异常

int CODE_LOAD_MFITEM = 73;//补丁SOPHIX.MF文件解析异常

int CODE_LOAD_COPY_FILE = 74;//补丁拷贝异常

int CODE_LOAD_SIGNATURE = 75;//补丁签名校验异常

int CODE_LOAD_SOPHIX_VERSION = 76;//补丁和补丁工具版本不一致异常

int CODE_LOAD_NOT_ZIP_FORMAT = 77;//补丁zip解析异常

int CODE_LOAD_DELETE_OPT = 80;//删除无效odex文件异常

int CODE_LOAD_HANDLE_DEX = 81;//加载dex异常

// 反射调用异常

int CODE_LOAD_FIND_CLASS = 82;

int CODE_LOAD_FIND_CONSTRUCTOR = 83;

int CODE_LOAD_FIND_METHOD = 84;

int CODE_LOAD_FIND_FIELD = 85;

int CODE_LOAD_ILLEGAL_ACCESS = 86;

//resource加载

public static final int CODE_LOAD_RES_ADDASSERTPATH = 123;//新增资源补丁包异常

//lib加载

int CODE_LOAD_LIB_UNDEFINED = 131;//未定义异常

int CODE_LOAD_LIB_CPUABIS = 132;//获取primaryCpuAbis异常

int CODE_LOAD_LIB_JSON = 133;//json格式异常

int CODE_LOAD_LIB_LOST = 134;//lib库不完整异常

int CODE_LOAD_LIB_UNZIP = 135;//解压异常

int CODE_LOAD_LIB_INJECT = 136;//注入异常

1.4 版本管理说明

说明一:patch是针对客户端具体某个版本的,patch和具体版本绑定

eg. 应用当前版本号是1.1.0, 那么只能在后台查询到1.1.0版本对应发布的补丁, 而查询不到之前1.0.0旧版本发布的补丁.

说明二:针对某个具体版本发布的新补丁, 必须包含所有的bugfix, 而不能依赖补丁递增修复的方式, 因为应用仅可能加载一个补丁

eg. 针对1.0.0版本在后台发布了一个补丁版本号为1的补丁修复了bug1, 然后发现此时针对这个版本补丁1修复的不完全, 代码还有bug2, 在后台重新发布一个补丁版本号为2的补丁, 那么此时补丁2就必须同时包含bug1和bug2的修复才行, 而不是只包含bug2的修复(bug1就没被修复了)

1.5 客户端本地SDK及DEMO下载

下载客户端SDK并集成(下载地址),Demo程序(Github)

1.6 注意事项

发布前请严格按照:扫码内测 => 灰度发布 => 全量发布的流程进行,以保证补丁包能够正常在所有Android版本的机型上生效。为了保险起见,理论上应该对每个版本的android手机都测一遍是否生效会比较好。不过,其实只需测试通过以下具有代表性的Android版本就基本没什么大问题了:4.0、4.4、5.1、7.0

遇到问题先查看文档:https://help.aliyun.com/knowledge_list/51422.html

1.7 补丁使用

生成的补丁需要上传到控制台。控制台地址

详情说明见:管理控制台使用

在阿里云平台创建App

image

创建完成后对应的 AndroidManifest的配置

android:name="com.taobao.android.hotfix.IDSECRET"

android:value="App ID" />

android:name="com.taobao.android.hotfix.APPSECRET"

android:value="App Secret" />

android:name="com.taobao.android.hotfix.RSASECRET"

android:value="RSA密钥" />

image.png
MyAppliction:
private void initHotfix() {
    String appVersion;
    try {
        appVersion = this.getPackageManager().getPackageInfo(this.getPackageName(), 0).versionName;
    } catch (Exception e) {
        appVersion = "1.0.0";
    }
    SophixManager.getInstance().setContext(this)
            .setAppVersion(appVersion)
            .setAesKey(null)
            .setEnableDebug(true)
            .setPatchLoadStatusStub(new PatchLoadStatusListener() {
                @Override
                public void onLoad(int mode, int code, String info, int handlePatchVersion) {
                    //补丁加载回掉通知
                    if (code == PatchStatus.CODE_LOAD_SUCCESS){
                        //表明补丁加载成功
                        Log.i("提示","表示补丁加载成功");
                    }else if(code == PatchStatus.CODE_LOAD_RELAUNCH){
                        //表明新补丁生效需要重启,开发者可以提示用户或者强制重启
                        // 建议: 用户可以监听进入后台事件, 然后调用killProcessSafely自杀,以此加快应用补丁,详见1.3.2.3
                    }else if(code == PatchStatus.CODE_LOAD_FAIL){
                        //清空本地所有补丁包
                        SophixManager.getInstance().cleanPatches();
                    }else{
                        //其他错误信息,查看PatchStatus类说明
                    }
                }
            }).initialize();
    // queryAndLoadNewPatch不可放在attachBaseContext 中,否则无网络权限,建议放在后面任意时刻,如onCreate中
    SophixManager.getInstance().queryAndLoadNewPatch();
}

Demo写的比较简单,主要就是为了快速测试一下
如果没有热更新,可能就要搞个临时版本或者甚至发布一个新版本,但是现在我们有了 Sophix ,就不需要这么麻烦了。

没补丁之前是这样的
image.png

image.png
更新补丁之后是这样的
image.png
image.png

首先我们去下载补丁打包工具(不得不说,工具确实比较粗糙(丑)。。。)

image

旧包:<必填> 选择基线包路径(有问题的 APK)。

新包:<必填> 选择新包路径(修复过该问题 APK)。

日志:打开日志输出窗口。

高级:展开高级选项

设置:配置其他信息。

GO!:开始生成补丁。

所以首先我们把旧包和新包添加上之后,配置好之后看看会发生什么吧!

强制冷启动是补丁打完后重启才生效。

image
image
image

时间看情况吧,因为项目本身内容比较少,所以生成补丁的速度比较快,等一下就好了。项目比较大的话估计需要等的时间长一点

我们来看看到底生成了什么?打开补丁生成目录

image

这个就是我们生成的补丁文件了,下一步补丁如何使用?

我们打开阿里的管理控制台,将补丁上传到控制台,就可以发布了.

image.png
image.png

我们首先下载调试工具来看看效果吧,首先连接应用(我用了真机和模拟器都没有出现过问题)

image.png

然后有两种方式可以加载补丁包,一种是扫二维码,还有一种是加载本地补丁jar包,这个模拟器不方便操作,我用的真机调试的

image.png

从图中的 log 提示我们可以看出首先下载了补丁包,然后打补丁完成,要求我们重启 APP,那我们就重启呗,看到的当然就应该是补丁打好的 1.0.1 版本和 Toast 弹出正常啦!!(这里有个坑,在阿里云的管理控制台,补丁需要与对应的版本进行更新,比如最新应用版本为1.0,你就在1.0版本上添加补丁,而不要在1.0.1版本上添加补丁,否则应用会找不到需要更新的补丁)

image.png
image.png

当然了,目前我们还是在调试工具上加载的补丁包,我们接下来将补丁包发布后就可以不用调试工具,直接打开 app 就可以实现打补丁了,这样就完成了 bug 的修复!

其实这么看起来确实是非常简单就实现了热修复,主要我们的生成补丁工作都是交给阿里提供的工具实现了,其实我们也能看得出来,Sophix 和前面介绍的 AndFix 很像,不同的地方是补丁文件已经给出工具可以一键生成了,而且支持的东西更多了。其他比如 so 库和 library 以及资源文件的更新大家可以查看官方文档了解。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,783评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,360评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 142,942评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,507评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,324评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,299评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,685评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,358评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,652评论 1 293
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,704评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,465评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,318评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,711评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,991评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,265评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,661评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,864评论 2 335

推荐阅读更多精彩内容