1.背景
当一个App发布之后,突然发现了一个严重bug需要进行紧急修复,这时候公司各方就会忙得焦头烂额:重新打包App、测试、向各个应用市场和渠道换包、提示用户升级、用户下载、覆盖安装。有时候仅仅是为了修改了一行代码,也要付出巨大的成本进行换包和重新发布。
这时候就提出一个问题:有没有办法以补丁的方式动态修复紧急Bug,不再需要重新发布App,不再需要用户重新下载,覆盖安装?
虽然Android系统并没有提供这个技术,但是很幸运的告诉大家,答案是:可以,我们QQ空间提出了热补丁动态修复技术来解决以上这些问题。
2.实际案例
空间Android独立版5.2发布后,收到用户反馈,结合版无法跳转到独立版的访客界面,每天都较大的反馈。在以前只能紧急换包,重新发布。成本非常高,也影响用户的口碑。最终决定使用热补丁动态修复技术,向用户下发Patch,在用户无感知的情况下,修复了外网问题,取得非常好的效果。
3.解决方案
该方案基于的是android dex
分包方案的,关于dex
分包方案,网上有几篇解释了,所以这里就不再赘述,具体可以看这里https://m.oschina.net/blog/308583。
简单的概括一下,就是把多个dex
文件塞入到app
的classloader
之中,但是android dex
拆包方案中的类是没有重复的,如果classes.dex
和classes1.dex
中有重复的类,当用到这个重复的类的时候,系统会选择哪个类进行加载呢?
让我们来看看类加载的代码:
一个ClassLoader
可以包含多个dex
文件,每个dex
文件是一个Element
,多个dex
文件排列成一个有序的数组dexElements
,当找类的时候,会按顺序遍历dex
文件,然后从当前遍历的dex
文件中找类,如果找类则返回,如果找不到从下一个dex
文件继续查找。
理论上,如果在不同的dex
中有相同的类存在,那么会优先选择排在前面的dex
文件的类,如下图:
在此基础上,我们构想了热补丁的方案,把有问题的类打包到一个dex(patch.dex)
中去,然后把这个dex
插入到Elements
的最前面,如下图:
好,该方案基于第二个拆分dex
的方案,方案实现如果懂拆分dex
的原理的话,大家应该很快就会实现该方案,如果没有拆分dex
的项目的话,可以参考一下谷歌的multidex
方案实现。然后在插入数组的时候,把补丁包插入到最前面去。
好,看似问题很简单,轻松的搞定了,让我们来试验一下,修改某个类,然后打包成dex
,插入到classloader
,当加载类的时候出现了(本例中是QzoneActivityManager
要被替换):
为什么会出现以上问题呢?
从log
的意思上来讲,ModuleManager
引用了QzoneActivityManager
,但是发现这这两个类所在的dex
不在一起,其中:
ModuleManager
在classes.dex
中QzoneActivityManager
在patch.dex
中
结果发生了错误。
这里有个问题,拆分dex
的很多类都不是在同一个dex
内的,怎么没有问题?
让我们搜索一下抛出错误的代码所在,嘿咻嘿咻,找到了一下代码:
从代码上来看,如果两个相关联的类在不同的dex中就会报错,但是拆分dex没有报错这是为什么,原来这个校验的前提是:
如果引用者(也就是ModuleManager
)这个类被打上了CLASS_ISPREVERIFIED
标志,那么就会进行dex
的校验。那么这个标志是什么时候被打上去的?让我们在继续搜索一下代码,嘿咻嘿咻~~,在DexPrepare.cpp
找到了一下代码:
这段代码是dex
转化成odex(dexopt)
的代码中的一段,我们知道当一个apk
在安装的时候,apk
中的classes.dex
会被虚拟机(dexopt)
优化成odex
文件,然后才会拿去执行。
虚拟机在启动的时候,会有许多的启动参数,其中一项就是verify
选项,当verify
选项被打开的时候,上面doVerify变量为true,那么就会执行dvmVerifyClass
进行类的校验,如果dvmVerifyClass
校验类成功,那么这个类会被打上CLASS_ISPREVERIFIED
的标志,那么具体的校验过程是什么样子的呢?
此代码在DexVerify.cpp中,如下:1
2
3
4
51. 验证clazz->directMethods方法,directMethods包含了以下方法:
1. static方法
2. private方法
3. 构造函数
2. clazz->virtualMethods虚函数=override方法?
概括一下就是如果以上方法中直接引用到的类(第一层级关系,不会进行递归搜索)和clazz都在同一个dex中的话,那么这个类就会被打上CLASS_ISPREVERIFIED:
所以为了实现补丁方案,所以必须从这些方法中入手,防止类被打上CLASS_ISPREVERIFIED标志。
最终空间的方案是往所有类的构造函数里面插入了一段代码,代码如下:
1 | if (ClassVerifier.PREVENT_VERIFY) { |
其中AntilazyLoad
类会被打包成单独的hack.dex
,这样当安装apk
的时候,classes.dex
内的类都会引用一个在不相同dex
中的AntilazyLoad
类,这样就防止了类被打上CLASS_ISPREVERIFIED
的标志了,只要没被打上这个标志的类都可以进行打补丁操作。
然后在应用启动的时候加载进来.AntilazyLoad
类所在的dex
包必须被先加载进来,不然AntilazyLoad
类会被标记为不存在,即使后续加载了hack.dex
包,那么他也是不存在的,这样屏幕就会出现茫茫多的类AntilazyLoad
找不到的log
。
所以Application
作为应用的入口不能插入这段代码。(因为载入hack.dex
的代码是在Application
中onCreate
中执行的,如果在Application
的构造函数里面插入了这段代码,那么就是在hack.dex
加载之前就使用该类,该类一次找不到,会被永远的打上找不到的标志)
其中:
之所以选择构造函数是因为他不增加方法数,一个类即使没有显式的构造函数,也会有一个隐式的默认构造函数。
空间使用的是在字节码插入代码,而不是源代码插入,使用的是javaassist库来进行字节码插入的。
隐患:
虚拟机在安装期间为类打上CLASS_ISPREVERIFIED
标志是为了提高性能的,我们强制防止类被打上标志是否会影响性能?这里我们会做一下更加详细的性能测试.但是在大项目中拆分dex
的问题已经比较严重,很多类都没有被打上这个标志。
如何打包补丁包:
空间在正式版本发布的时候,会生成一份缓存文件,里面记录了所有class文件的md5,还有一份mapping混淆文件。
在后续的版本中使用-applymapping选项,应用正式版本的mapping文件,然后计算编译完成后的class文件的md5和正式版本进行比较,把不相同的class文件打包成补丁包。
备注:该方案现在也应用到我们的编译过程当中,编译不需要重新打包dex,只需要把修改过的类的class文件打包成patch dex,然后放到sdcard下,那么就会让改变的代码生效。
转载自 Android 热修复其实很简单
一、什么是热修复
热修复说白了就是”打补丁”,比如你们公司上线一个app,用户反应有重大bug,需要紧急修复。如果按照通常做法,那就是程序猿加班搞定bug,然后测试,重新打包并发布。这样带来的问题就是成本高,效率低。于是,热修复就应运而生.一般通过事先设定的接口从网上下载无Bug的代码来替换有Bug的代码。这样就省事多了,用户体验也好。
二、热修复的原理
- Android的类加载机制
Android的类加载器分为两种,PathClassLoader
和DexClassLoader
,两者都继承自BaseDexClassLoader
PathClassLoader
代码位于libcore\dalvik\src\main\java\dalvik\system\PathClassLoader.java
DexClassLoader
代码位于libcore\dalvik\src\main\java\dalvik\system\DexClassLoader.java
BaseDexClassLoader
代码位于libcore\dalvik\src\main\java\dalvik\system\BaseDexClassLoader.java
PathClassLoader
用来加载系统类和应用类
DexClassLoader
用来加载jar、apk、dex
文件.加载jar、apk
也是最终抽取里面的Dex
文件进行加载.
这里写图片描述
- 热修复机制
看下PathClassLoader
代码
1 | public class PathClassLoader extends BaseDexClassLoader { |
DexClassLoader
代码
1 | public class DexClassLoader extends BaseDexClassLoader { |
两个ClassLoader
就两三行代码,只是调用了父类的构造函数.
1 | public class BaseDexClassLoader extends ClassLoader { |
在BaseDexClassLoader 构造函数中创建一个DexPathList类的实例,这个DexPathList的构造函数会创建一个dexElements 数组
1 | public DexPathList(ClassLoader definingContext, String dexPath, String libraryPath, File optimizedDirectory) { |
然后BaseDexClassLoader 重写了findClass方法,调用了pathList.findClass,跳到DexPathList类中.
1 | /* package */final class DexPathList { |
会遍历这个数组,然后初始化DexFile,如果DexFile不为空那么调用DexFile类的loadClassBinaryName方法返回Class实例.
归纳上面的话就是:ClassLoader会遍历这个数组,然后加载这个数组中的dex文件.
而ClassLoader在加载到正确的类之后,就不会再去加载有Bug的那个类了,我们把这个正确的类放在Dex文件中,让这个Dex文件排在dexElements数组前面即可.
这里有个问题,可参考QQ空间团队的 安卓App热补丁动态修复技术介绍
概括来讲:如果引用者和被引用者的类(直接引用关系)在同一个Dex时,那么在虚拟机启动时,被引用类就会被打上CLASS_ISPREVERIFIED标志,这样被引用的类就不能进行热修复操作了.
那么我们就要阻止被引用类打上CLASS_ISPREVERIFIED标志.QQ空间的方法是在所有引用到该类的构造函数中插入一段代码,代码引用到别的类.
三、热修复的例子
其实它的原理也是动态加载class
文件,然后调用反射完成修复.可参考我上一篇写的
Java的ClassLoader加载机制
AndFix
是“Android Hot-Fix”
的缩写。它支持Android 2.3到6.0版本,并且支持arm与X86系统架构的设备。完美支持Dalvik与ART的Runtime。AndFix 的补丁文件是以.apatch
结尾的文件。
我这是用eclipse
写的Demo
.
- 把AndFix抽取成library依赖的形式
这里写图片描述
- 新建一个AndFixDemo项目,依赖AndFix这个library
2.1
新建一个MyApplication继承Application
1 | public class MyApplication extends Application { |
实际当中肯定是通过网络接口下载apatch
文件,我这里为了方便演示就放在了SD
卡根目录
2.2
在MainActivity
用一个按钮弹出吐司,上面是有Bug
的代码,下面是修正后的代码
这里写图片描述
这里写图片描述
分别打包成Bug.apk和NoBug.apk
这里写图片描述
2.3
然后要用到一个生成补丁的工具apkpatch
解压
这里写图片描述
_MACOSX是给OSX系统用的
.bat是给window系统用的
我用得是.bat
把之前生成的Bug.apk
和NoBug.apk
,还有打包所使用的keystore
文件放到apkpatch-1.0.3
目录下
打开cmd
,进入到apkpatch-1.0.3
目录下,输入如下指令
apkpatch.bat -f NoBug.apk -t Bug.apk -o Dennis -k keystore -p 111111 -a 111111 -e 111111
每个参数含义如下
1 | -f 新版本的apk |
这里写图片描述
如果出现add modified …….就表示成功了,去apkpatch-1.0.3目录看下,新增了Dennis目录
这里写图片描述
这里写图片描述
我把这个文件改为Dennis.apatch
2.4
手机装上Bug.apk运行起来
这里写图片描述
然后把Dennis.apatch
放到SD
卡根目录,退出app
,再进入,按下按钮
这里写图片描述
最后附上Demo还有apk和apatch 文件 打开链接
Nuwa
Nuwa is a goddess in ancient Chinese mythology best known for repairing the pillar of heaven.With this Nuwa project,you can also have the repairing power, fix your android applicaiton without have to publish a new APK to the appstore.