很久以前就有想了解熱修复的问题,但是很多情况下都缺乏实践,网上看了很多hook的理论和实践,近来研究了一番发现现在比较多使用的是阿里的AndFix,他能兼容android2.3~6.0的热修复,而且修复速度非常快,补丁也非常小,用于修复紧急性bug是非常有效的。
AndFix最重要是运用到(方法的替换),有些热修复框架是将类替换,有些则替换dex
1.实现的方法替换
2.实例演示
当我们下载了github的demo之后,可以看到demo仲提供了源码和example
(1)import exmple的工程
(2)加入AndFix的库
(3)然后我们需要修改MainActivity,添加一个TextView显示
(4)然后要生成签名打包一个apk,然后打包改名为1.apk
(5)修改TextView的文本
(6)编译打包成新的apk,改名为2.apk
(7)demo里面tools的文件夹里面,有apkpatch的工具
需要使用它做一个补丁包
旧的apk为1.apk,新的apk为2.apk, -o表示补丁的输出目录,-k表示keystore, -p表示keystore的密码,-a表示alias, -e表示entry password
然后会成一个.apatch的补丁包
(8)将.apatch的补丁包改名为out.apatch,然后放到手机的主目录。
查看一下MainApplication
可以看到MainApplication启动的时候会加载PatchManager,然后他会默认加载一个out.apatch的补丁文件。
(9)再次进入app就会发现textview会显示bug fix证明更新是成功的
3.补丁命名的规则
a_b.apatch
a表示versionCode,b表示当前的补丁的版本。
例如:如果当前的versionCode的版本为4,补丁的版本为3,则命名为4_3.apatch。
4.适用性
Andfix并不能修复所有情况下出现的bug,测试结果如下:
其中对类得构造方法和Application的onCreate方法,还有的xml绑定方法是不支持的
5.补丁加载的时机
一般情况下是越早加载越好,一般在MainApplication启动的onCreate方法启动补丁下载,然后调用pathManager的loadPatch方法
1.可以考虑下下载补丁,再下一次启动后加载
2.或者下载补丁,立刻加载,然后会重启app
6.补丁替换内部流程
1.补丁下载到相应目录
2.通过loadPatch方法加载
3.会将补丁复制到app的目录里面,然后下载补丁的文件会被删除
4.如果versionName改变了,将会删除apatch里面所有的补丁。
7.混淆
-printmapping proguard.map
首先需要生成mapping文件记录混淆规则,之后可以把printmapping 这句话注释掉,每次只使用applymapping。
-applymapping proguard.map
然后在下面加上
-keepclass*extendsjava.lang.annotation.Annotation
-keepclasseswithmembernamesclass*{native ;}
-keepclasscom.alipay.euler.andfix.**{ *; }
8.局限性
1.无法添加新类和新字段
2.无法替换构造函数和Application的onCreate方法
3.未测试MVVM绑定和注解方法是否可以实现替换。
4.无法动态加入新功能模块,有别于dex的替换
参考资料