Android手写热修复(一)--ClassLoader

前言

在上一篇文章Android类加载机制讲解了类加载器、加载dex、查找class相关的内容,并且透漏了热修复的原理,还没有看过的同学建议先看上一篇再来学习本文。

热修复的几种方案

1、基于类加载机制
2、底层替换方法
3、instant run 方法

今天我们研究的是方案1,其他方案以及资源文件修复后面文章会说。

基于类加载的修复原理

1、首先Android的类加载也是基于双亲委托机制,一个类只会被加载一次。那么我可以把有bug的类打包成补丁dex,下发到手机中,然后在加载原来的bug类之前加载补丁dex中修复好的类,那以后就不会再加载原来的bug类了,这种思路就可以到达修复class文件bug的目的。

2、思路很明确了,打包补丁dex也很容易(具体操作后面会说),下发补丁到app也容易。那就只剩下一个问题了,如何让已经修复好的类先于有bug的类加载?

在上一篇文章,我们对查找class做了较为详细的分析。其实答案就包含在其中,这里再回顾一下。查找class是通过DexPathList来完成的,内部是DexPathList遍历Element数组,通过Element获取DexFile对象来加载Class文件。由于数组是有序的,如果2个dex文件中存在相同类名的class,那么类加载器就只会加载数组前面的dex中的class。如果apk中出现了有bug的class,那只要把修复的class打包成dex文件并且放在DexPathList中Element数组的前面,就可以实现bug修复了。

热修复--类加载.png

关键步骤:
1、要获取加载apk的dex和布丁dex的类加载器PathClassLoader、DexClassLoader;
2、要获取Elements[]数组,只能通过反射BaseDexClassLoader、DexPathList这2个类去获取;
3、获取到2个dex文件的Element数组之后,需要创建一个新的数组把这2个数组合并,补丁dex的数组放在前面,原apk中放在后面;
4、再通过反射找去替换掉原来的dexElements属性即可。

类加载修复的实现

以下是工具类中的核心代码,需要完整代码的同学去文末git地址获取

   /**
     * 尝试加载app私有目录中的补丁dex去修复bug
     * @param context
     */
    fun loadPatch(context: Context) {
        //这个是补丁dex存放的位置
        dexPath = context.filesDir.path + "/patch0.dex"
        if (!File(dexPath).exists())
            return
        //注意,只能用app私有目录去存放优化后的dex文件,如果放在外部存储存在注入攻击的风险
        // data/data/包名/files/opt_dex
        //在8.0及这个参数没有作用,只能用系统指定位置
        optPath = context.filesDir.path + "/opt_dex"
        var optFile = File(optPath)
        if (!optFile.exists()) {
            optFile.mkdirs()
        }

        //第一步,获取/创建apk和补丁dex的类加载器
        var pathClassLoader: PathClassLoader = context.classLoader as PathClassLoader
        var dexClassLoader = DexClassLoader(dexPath, optPath, null, pathClassLoader)
        //第二步,反射获取BaseDexClassLoader中的DexPathList pathList属性
        var pathPathList = getPathList(pathClassLoader)
        var dexPathList = getPathList(dexClassLoader)
        //第三步,反射获取获取DexPathList中的Element[] dexElements数组;
        var pathElements = getDexElements(pathPathList)
        var dexElements = getDexElements(dexPathList)
        //第四步,合并Elements数组,注意补丁dex的数组要放在前面
        var combineElements = combineArray(dexElements, pathElements)
        //第五步,重新给PathClassLoader中的Element[] dexElements赋值(其实在PathList里面)
        var pathList = getPathList(pathClassLoader) //再次获取apk中的PathList对象
        setField(pathList, pathList.javaClass, DEX_ELEMENTS_FIELD, combineElements)
    }

加载补丁dex

class MyApplication : Application() {
    override fun onCreate() {
        super.onCreate()
        Log.e("tag", "Application  onCreate: ")
        loadPatch()
    }

    /**
     * 1、这里测试是直接把补丁dex放在data/data目录,实际开发过程中可以把dex文件放在服务器,
     * 在特定时机通过网络下载到app指定目录,然后在Application加载补丁dex
     *
     * 2、第二次进入的时候可以根据目录下是否已经下载过,处理,避免重新下载
     *
     */
    private fun loadPatch() {
//        var dexFilePath: String = filesDir.path + "/classes.dex"
        HotFixEngine.loadPatch(this)
        var intent = Intent(this, MainActivity::class.java)
        startActivity(intent)
    }
}

加载补丁需要注意的2点,在注释已经标注了,主要是加载时机和避免重复加载。

注意,我这里是手动把补丁dex放在特定目录,我的目录是data/data/包名/files/patch0.dex,如果你要运行加载补丁的代码,你需要先把补丁dex放在特定目录,然后修改dexPath指向的路径才可以。

关于class文件打包dex,用android sdk tools自带的工具就可以了,具体操作可以参考: Android studio .class文件手动生成dex

项目地址:
https://github.com/zhouxu88/HotFix

参考

Android 类加载机制及热修复原理
热修复——深入浅出原理与实现
从JVM到Dalivk再到ART(class,dex,odex,vdex,ELF)

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

推荐阅读更多精彩内容