一步一步学习Android热修复技术
一、Dex
java → class
javac Hello.java
class → dex
dx --dex --output Hello.dex Hello.class
adb
adb push Hello.dex /storage/emulated/0
adb shell
dalvikvm -cp /sdcard/Hello.dex Hello (执行dex中的哪个类)
二、ClassLoader
1.Java
2.Android
BootClassLoader:加载Android Framework层的class字节码文件
PathClassLoader:与Java中的AppClassLoader类似,用来加载已经安装的系统中的apk文件中的class文件
DexClassLoader:与Java中的CustomClassLoader类似,用来加载指定目录中的class字节码文件
BaseDexClassLoader:PathClassLoader和DexClassLoader的父类
运行一个APP至少需要BootClassLoader和PathClassLoader
双亲代理模型的特点:
当前ClassLoader是否已加载过此类,如果已经加载过则直接返回,否则会去查询Parent是否已经加载此类,如果已经加载过则直接返回Parent加载过的这个类,如果整个继承线路上的ClassLoader都没有加载过这个类,当前的ClassLoader才会去加载这个类。
类加载的共享功能:
一个类被顶层的ClassLoader加载过就会缓存在内存里,以后不管在哪里用到都不会重新加载。
类加载的隔离功能:
不同继承路线下的ClassLoader加载的类,肯定不是同一个类,避免非法篡改代码访问核心类库中的成员变量。
见ClassLoader类中的loadClass方法
*动态加载难点
1.有许多组件类需要注册才能使用
2.资源的动态加载很复杂
总结:Android程序运行需要一个上下文环境。
热修复
AndFix
https://github.com/alibaba/AndFix
BUILD (需要配置签名)
AS Terminal → ./gradlew assembleRelease
apkpatch
生成apatch文件
usage: apkpatch -f -t -o -k -p <***> -a -e <***>
-a,--alias keystore entry alias.
-e,--epassword <***> keystore entry password.
-f,--from new Apk file path.
-k,--keystore keystore path.
-n,--name patch name.
-o,--out output dir.
-p,--kpassword <***> keystore password.
-t,--to old Apk file path.
合并apatch文件
usage: apkpatch -m -o -k -p <***> -a -e <***>
-a,--alias keystore entry alias.
-e,--epassword <***> keystore entry password.
-k,--keystore keystore path.
-m,--merge path of .apatch files.
-n,--name patch name.
-o,--out output dir.
-p,--kpassword <***> keystore password.
接下来需要实现的就是通过代码将apatch文件下载到用户手机中 然后通过AndFix API中的方法去修复就可以了。
Tinker
主要包括一下几个部分:
1.gradle编译插件:tinker-patch-gradle-plugin
帮助我们在AS中直接完成patch文件生成
2.核心sdk库:tinker-android-lib
tinker提供的API
3.非gradle编译用户的命令行版本:tinker-patch-cli.jar
为eclipse而生 2333
原理
1.基于android原生的Class Loader,开发了自己的Class Loader,加载patch文件中的字节码
2.基于android原生的aapt,开发了自己的aapt
3.基于Dex文件格式,研发了DexDiff算法进行比对生成patch文件
使用方式
//optional, help to generate the final application
//生成application时使用
provided('com.tencent.tinker:tinker-android-anno:1.9.1')
//tinker's main Android lib
//tinker的核心sdk库
compile('com.tencent.tinker:tinker-android-lib:1.9.1')
(provided仅编译时生效,不会被打包进apk)