Android 热修复思路整理

Android 热修复思路整理

问题提出

  • 什么是热修复
  • 热修复的优点
  • 流行的热修复方式
  • 本文采用的热修复
    • 方式
    • 相关知识点
    • 思路

解决方案

  • 热修复:

    • Android端修复bug,通常方案是,修改bug所在的类->打包整个代码apk->重新上线->用户下载apk重新安装使用;
    • 热修复的方式,修改bug所在的类->用户在已有app的情况下下载修复好的fixbug.class。
  • 热修复的优点:很明显开发人员不需要重新打包整个apk重新上传apk,用户不需要重新下载整个apk,在用户使用已有app的时候,偷偷的将bug就修复了,用户甚至不知道发生过什么,提高用户体验。

  • 流行的热修复方式(按技术特点划分)

    • 通过底层二进制(C/C++)方式,阿里:DeXposed。andfix。
    • 通过JAVA层加载机制, 腾讯:tinker
  • 本文采用的热修复

    • 比较容易理解的Java类加载机制实现的。
    • 主要用到的技术:JAVA反射,热修复的热字就体现在这里,运行时fix bug...
  • 思路
    • 以前搞过反编译的小伙伴已经反编译过classes.dex文件,classes文件是如何生成的呢?这就好办了找到生成dex文件的地方就可以了。。。
    • 其实JAVA类的加载机制是从:dalvik.system.BaseDexClassLoader开始的,查看源码我们发现里面使用了DexPathList这个类,这个类里面的是通过dexElements数组,来处理我们反编译出来的classes.dex, classes2.dex文件
    • 如果在细看代码你会发现,已经加载过的.class文件就不会再被加载了,也就是说同一时刻ClassLoder只会加载一次同一个.class文件一次,那我们是不是可以在有问题的.class文件加在之前,先加载我们的修复好的.class文件是不是就可以了。
    • 那剩下的问题就剩下如何获取到dexElements数组,将修复好的dex合并到dexElements数组里面,重新让系统使用即可。
    • 那我们先来确认dex文件位于app那个位置呢,答案就是在当前app的getDir()目录下(data\data...),我们只要将修改好的java文件生成.class文件在编译成classes.dex文件再将该文件存入到该目录下,通过反射获取我们的classes.dex进行数组合并工作,就可以达到目的,下面着重说一下反射思路。(生成classes.dex文件可用:你的SDK目录\build-tools\24.0.3\dx.bat工具)

- 反射思路:

 //加载应用程序的ClassLoder
    ClassLoader classLoader = context.getClassLoader();
    for (File file : loadedDex) {
        //因为系统通过DexClassLoder来加载dex,所以需要将新加入的修复dex伪装成系统能够识别的ClassLoder
        DexClassLoader dexClassLoder = new DexClassLoader(
                file.getAbsolutePath(),// String dexPath,
                optFile.getAbsolutePath(),// String optimizedDirectory,
                null,//String libraryPath,
                classLoader// ClassLoader parent
        );
       //分别通过系统和Fix Dex反射获取DexClassLoader对象的DexPathList对象
        // 进一步获取DexPathList对象的dexElements数组对象
        //将两个数组利用已经loaderd的类不会呗再次加载的特点,将fix数组放在dexElements数组下标0处,合并两个数组
        //通过放射将修改后的dexElements数组重新写入到app加载类的目录下,达到修复的目的,此时错误的class文件还是存在与数组内部的,但是已经不会再被加载了
  • 可能遇到的坑

    • 需要将AS的Instant Run功能暂时关闭,不然修复的内容有可能无法写入到classes.dex看不到效果
    • 需要配置APP的gradle里的mutidex

你可能感兴趣的:(Android进阶)