美文网首页Android开发Android开发经验谈Android技术知识
一文带你手动实现最简单的Android热修复

一文带你手动实现最简单的Android热修复

作者: 4ca1bbef6a0c | 来源:发表于2020-05-29 12:05 被阅读0次

原作者:黄庆庆 来源CSDN
原文链接:https://blog.csdn.net/hq942845204/article/details/81044158

前言

最近了解到了热修复相关的东西,于是很好奇原理,便一番搜索资料,同时为了加深对热修复的理解,便自己照着网上的例子去实现一个热修复,因为基础相对比较差,而且网上很多例子都是过时的,而且很多细节不注意到的话,就是一个坑,而且还五花八门的,于是我觉得将自己的这个实现热修复的例子记录下来是很有必要的,主要是参考并综合了网上很多热修复的例子,自己实现并完成整个从0到1的过程,好了,我们开始吧!

需要知道的基本概念和原理

首先是热修复的基本概念,我不太喜欢那种专业术语的描述方式,因为那样很容易让人觉得晦涩难懂,而且我觉得唯一的效果就是营造出一种初学者觉得高大上的装X效果而已,所以我就说下我的理解:假定一个场景,你的APP上线了,现在发现了一个小Bug,这个Bug很简单,可能是一行代码的事,但是由于你才上线,要是再重新打包你的APP再上线,这个过程就很麻烦了,于是我们期望有一种方式,不需要用户来重新安装新的APP即可运行我们修复了Bug的APP,这种方式就叫热修复。

是不是觉得很神奇。我也是,在没接触之前,我也觉得很神奇,但是明白之后,其实真的没什么,很简单。

再说下热修复的基本原理,这里很多网上的讲解非常的专业,我看了之后也理解了好久,但是自己再梳理一下,其实没有那么难理解,我还是以通俗的方式来说:

假设有一个数组,这个数组,里面有很多个dex文件(不了解的只需要知道里面就是存放了类的二进制数据,用来给安卓虚拟机加载),然后安卓虚拟机在加载类文件的时候,会有个顺序,我们暂且不用管是什么顺序,或者是怎么加载的,我们只需要知道,它会有顺序,我们假定它从数组下标为0的地方开始循环找,一旦找到了对应的文件,那么后面即便仍然还有该类的dex文件,也不去加载了,相当于前面加载的会覆盖后面的,就是这样一个原理。

那么实际中,可以怎么实现呢,我们可以将相应的dex文件放在服务器上,然后在用户不知道的情况下(可以在欢迎界面扫描服务器上的文件,如果有则下载进行热修复,否则不管),将这个dex文件从服务器上下载下来,并移动到指定位置即可。

我们也不需知道具体移动到哪里了,只知道移动的地方需要满足的条件是:在对应的类的dex文件加载顺序之前。这样就可以实现覆盖效果,让新的类文件比旧的类文件先加载,旧的就不会生效,达到我们想要的效果。

动手实现

在动手实现前,需要知道的事:

上面我们说了一种方案是从服务器上下载对应的dex文件,这里因为只是模拟一下效果,便采用手动复制对应的dex文件到指定目录下,来达到同样的效果。

开始吧:

首先我们新建工程,随便写一个Bug,比如我这里除数为0的Bug

public class TestCaculate {
    public int a = 10;
    public int b = 0;
    public void caculate(Context context) {
        Toast.makeText(context, "结果" + a / b, Toast.LENGTH_SHORT).show();
    }
}

当我们调用caculate方法时肯定会提示异常导致退出,现在我们以热修复的方式来修复Bug。
首先我们需要生成的文件就是我们修复好Bug的程序的dex文件,看清楚了,是修复好Bug的,代表什么意思呢,也就是在进行下一步之前,TestCaculate代码是这样的

public class TestCaculate {
    public int a = 10;
    public int b = 1;//已经修复
    public void caculate(Context context) {
        Toast.makeText(context, "结果" + a / b, Toast.LENGTH_SHORT).show();
    }
}

然后我们在布局文件中添加二个按钮,一个按钮点击调用caculate方法,触发Bug,另一个按钮点击修复Bug,需要注意的是,千万不要忘记了权限的申请,因为整个过程涉及到文件的读取和写入,而6.0以上需要动态获取权限,所以要在清单文件中加入下列两行代码。

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/></pre>

MainActivity代码如下

public class MainActivity extends AppCompatActivity {
    private Button btn, btn_fix;
    public static final int REQUEST_CODE = 1;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        btn = findViewById(R.id.btn);
        btn_fix = findViewById(R.id.btn_fix);
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                TestCaculate testCaculate = new TestCaculate();
                testCaculate.caculate(MainActivity.this);
            }
        });
        btn_fix.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                fix();
            }
        });
        ActivityCompat.requestPermissions(MainActivity.this,
                new String[]{Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUEST_CODE);
    }
    private void fix() {
        try {
            String dexPath = Environment.getExternalStorageDirectory() + "/classes2.dex";
            HotFix.patch(this, dexPath, "com.aiiage.testhotfix.TestCaculate");
            Toast.makeText(this, "修复成功", Toast.LENGTH_SHORT).show();
        } catch (Exception e) {
            Toast.makeText(this, "修复失败" + e.getMessage(), Toast.LENGTH_SHORT).show();
            e.printStackTrace();
        }
    }
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        switch (requestCode) {
            case REQUEST_CODE: {
                if (grantResults.length > 0) {
                    // permission was granted
                } else {
                    // permission denied
                }
                return;
            }
        }
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }
}

然后就是我们热修复的工具类,怎么使用,在MainActivity中已经有使用的代码了,工具类中用到了反射的知识,但是不是本文的重点,有需要的小伙伴自行学习相关知识,这个工具类中,最重要的两个类就是DexClassLoader和PathClassLoader,看名字就知道这是两个类加载器,用来加载类的,想知道具体实现的,下面就是源码

public final class HotFix {
    /**
     * 修复指定的类
     *
     * @param context        上下文对象
     * @param patchDexFile   dex文件
     * @param patchClassName 被修复类名
     */
    public static void patch(Context context, String patchDexFile, String patchClassName) {
        if (patchDexFile != null && new File(patchDexFile).exists()) {
            try {
                if (hasLexClassLoader()) {
                    injectInAliyunOs(context, patchDexFile, patchClassName);
                } else if (hasDexClassLoader()) {
                    injectAboveEqualApiLevel14(context, patchDexFile, patchClassName);
                } else {
                    injectBelowApiLevel14(context, patchDexFile, patchClassName);
                }
            } catch (Throwable th) {
            }
        }
    }
    private static boolean hasLexClassLoader() {
        try {
            Class.forName("dalvik.system.LexClassLoader");
            return true;
        } catch (ClassNotFoundException e) {
            return false;
        }
    }
    private static boolean hasDexClassLoader() {
        try {
            Class.forName("dalvik.system.BaseDexClassLoader");
            return true;
        } catch (ClassNotFoundException e) {
            return false;
        }
    }
    private static void injectInAliyunOs(Context context, String patchDexFile, String patchClassName)
            throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException,
            InstantiationException, NoSuchFieldException {
        PathClassLoader obj = (PathClassLoader) context.getClassLoader();
        String replaceAll = new File(patchDexFile).getName().replaceAll("\\.[a-zA-Z0-9]+", ".lex");
        Class cls = Class.forName("dalvik.system.LexClassLoader");
        Object newInstance =
                cls.getConstructor(new Class[]{String.class, String.class, String.class, ClassLoader.class}).newInstance(
                        new Object[]{context.getDir("dex", 0).getAbsolutePath() + File.separator + replaceAll,
                                context.getDir("dex", 0).getAbsolutePath(), patchDexFile, obj});
        cls.getMethod("loadClass", new Class[]{String.class}).invoke(newInstance, new Object[]{patchClassName});
        setField(obj, PathClassLoader.class, "mPaths",
                appendArray(getField(obj, PathClassLoader.class, "mPaths"), getField(newInstance, cls, "mRawDexPath")));
        setField(obj, PathClassLoader.class, "mFiles",
                combineArray(getField(obj, PathClassLoader.class, "mFiles"), getField(newInstance, cls, "mFiles")));
        setField(obj, PathClassLoader.class, "mZips",
                combineArray(getField(obj, PathClassLoader.class, "mZips"), getField(newInstance, cls, "mZips")));
        setField(obj, PathClassLoader.class, "mLexs",
                combineArray(getField(obj, PathClassLoader.class, "mLexs"), getField(newInstance, cls, "mDexs")));
    }
    @TargetApi(14)
    private static void injectBelowApiLevel14(Context context, String str, String str2)
            throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException {
        PathClassLoader obj = (PathClassLoader) context.getClassLoader();
        DexClassLoader dexClassLoader =
                new DexClassLoader(str, context.getDir("dex", 0).getAbsolutePath(), str, context.getClassLoader());
        dexClassLoader.loadClass(str2);
        setField(obj, PathClassLoader.class, "mPaths",
                appendArray(getField(obj, PathClassLoader.class, "mPaths"), getField(dexClassLoader, DexClassLoader.class,
                        "mRawDexPath")
                ));
        setField(obj, PathClassLoader.class, "mFiles",
                combineArray(getField(obj, PathClassLoader.class, "mFiles"), getField(dexClassLoader, DexClassLoader.class,
                        "mFiles")
                ));
        setField(obj, PathClassLoader.class, "mZips",
                combineArray(getField(obj, PathClassLoader.class, "mZips"), getField(dexClassLoader, DexClassLoader.class,
                        "mZips")));
        setField(obj, PathClassLoader.class, "mDexs",
                combineArray(getField(obj, PathClassLoader.class, "mDexs"), getField(dexClassLoader, DexClassLoader.class,
                        "mDexs")));
        obj.loadClass(str2);
    }
    private static void injectAboveEqualApiLevel14(Context context, String str, String str2)
            throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException {
        PathClassLoader pathClassLoader = (PathClassLoader) context.getClassLoader();
        Object a = combineArray(getDexElements(getPathList(pathClassLoader)),
                getDexElements(getPathList(
                        new DexClassLoader(str, context.getDir("dex", 0).getAbsolutePath(), str, context.getClassLoader()))));
        Object a2 = getPathList(pathClassLoader);
        //新的dexElements对象重新设置回去
        setField(a2, a2.getClass(), "dexElements", a);
        pathClassLoader.loadClass(str2);
    }
    /**
     * 通过反射先获取到pathList对象
     *
     * @param obj
     * @return
     * @throws ClassNotFoundException
     * @throws NoSuchFieldException
     * @throws IllegalAccessException
     */
    private static Object getPathList(Object obj) throws ClassNotFoundException, NoSuchFieldException,
            IllegalAccessException {
        return getField(obj, Class.forName("dalvik.system.BaseDexClassLoader"), "pathList");
    }
    /**
     * 从上面获取到的PathList对象中,进一步反射获得dexElements对象
     *
     * @param obj
     * @return
     * @throws NoSuchFieldException
     * @throws IllegalAccessException
     */
    private static Object getDexElements(Object obj) throws NoSuchFieldException, IllegalAccessException {
        return getField(obj, obj.getClass(), "dexElements");
    }
    private static Object getField(Object obj, Class cls, String str)
            throws NoSuchFieldException, IllegalAccessException {
        Field declaredField = cls.getDeclaredField(str);
        declaredField.setAccessible(true);//设置为可访问
        return declaredField.get(obj);
    }
    private static void setField(Object obj, Class cls, String str, Object obj2)
            throws NoSuchFieldException, IllegalAccessException {
        Field declaredField = cls.getDeclaredField(str);
        declaredField.setAccessible(true);//设置为可访问
        declaredField.set(obj, obj2);
    }
    //合拼dexElements
    private static Object combineArray(Object obj, Object obj2) {
        Class componentType = obj2.getClass().getComponentType();
        int length = Array.getLength(obj2);
        int length2 = Array.getLength(obj) + length;
        Object newInstance = Array.newInstance(componentType, length2);
        for (int i = 0; i < length2; i++) {
            if (i < length) {
                Array.set(newInstance, i, Array.get(obj2, i));
            } else {
                Array.set(newInstance, i, Array.get(obj, i - length));
            }
        }
        return newInstance;
    }
    private static Object appendArray(Object obj, Object obj2) {
        Class componentType = obj.getClass().getComponentType();
        int length = Array.getLength(obj);
        Object newInstance = Array.newInstance(componentType, length + 1);
        Array.set(newInstance, 0, obj2);
        for (int i = 1; i < length + 1; i++) {
            Array.set(newInstance, i, Array.get(obj, i - 1));
        }
        return newInstance;
    }
}

布局文件就两个按钮,就不贴了,占空间,好了,代码准备完毕,接着下一步吧。
接下来,我们生成项目对应的dex文件,网上资料有点少,,而且有的还是错的,各种莫名其妙的操作,哎说多了都是泪,但是也还是有正确的,我这里采用了一种相对简单的方式,首先在app的module下的build.gradle文件中加入代码,不要加入到某个节点下。最终代码如下

apply plugin: 'com.android.application'
android {
    compileSdkVersion 28
    defaultConfig {
        applicationId "com.aiiage.testhotfix"
        minSdkVersion 26
        targetSdkVersion 28
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}
dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation 'com.android.support:appcompat-v7:28.0.0-alpha3'
    implementation 'com.android.support.constraint:constraint-layout:1.1.2'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
}
//加入的代码从这里开始
task clearJar(type: Delete) {
    delete('libs/log.jar')
}
task makeJar(type: org.gradle.api.tasks.bundling.Jar) {
    //指定生成的jar名称
    baseName 'log'
    //从哪里打包class文件
    from('build/intermediates/classes/debug/com/aiiage/testhotfix/')
    //打包到jar后的目录结构
    into('com/aiiage/testhotfix/')
    //去掉不需要打包的目录和文件
    exclude('text/', 'BuildConfig.class', 'R.class', 'BuildConfig.class')
    exclude {
        it.name.startsWith('R/pre>);
    }
}
makeJar.dependsOn(clearJar, build)

加入的代码代表什么意思注释已经很清楚了,这个过程最终会生成一个jar包,然后打开AndroidStudio底下的命令行,如图

在命令行中,我们输入gradlew makeJar 注意,不要输错了,等待约2分钟左右,看到如下的字样,代表生成jar成功

生成的jar包存放的地方在配置文件中配置了,比如我这里就在这个目录下,如图

我这里的,名字叫log,所以最终得到的是一个名为log.jar的文件,现在我们用这个jar来得到dex文件,需要用到的工具是dx,这个工具在哪里呢,就是SDK目录下的build-tools,然后随便选择一个版本进去就可以看到名为dx.bat的文件,这个就是我们需要使用的。

我们将log.jar文件复制到这个目录下,按住shift右击鼠标在该目录下打开命令行,输入命令

dx --dex --output=D:/test log.jar</pre>

其中D:/test为保存生产的dex文件的目录,同时注意空格。回车若没有错误说明生产成功,我们来到指定的D:/test目录,发现我们的最终目标正静静的躺在里面等着我们,嘿嘿!

好了,我们现在将这个乖巧的classes.dex文件复制到我们的手机目录下,这里为了演示效果,我就采用的模拟器,如下,我这里将它重命名为classes2.jar,不重命名也没关系,名字无所谓,复制的目录为

这个过程在实际当中就是用户下载服务器上的对应文件,然后用代码将其放到指定目录下,只不过这里我们是手动模拟的这个操作。
然后我们就可以运行我们的程序了,但是运行程序之前还有两件事:
一:没猜错的话,你现在的代码是修复Bug后的代码,所以我们要将代码改会错误的版本,也就是下面这个

这样我们才能有Bug来修复嘛,不然我们Bug都没有,修复啥呢,对不
二:打开AndroidStudio的设置,取消掉instant run这里的勾勾

这样做是干嘛,取消掉这个勾勾之后,AndroidStudio在给我们安装新应用时,就不会只安装修改的部分,而是全部代码都重新编译并安装。

好了,我们准备工作做完了。接下来运行看效果吧。

首先我们点击修复按钮进行模拟热修复,看到修复成功的字样,说明修复成功,然后我们再点击HELLO按钮,这里按照预期会导致除数为0的异常,但是你会惊讶的发现,程序没有崩掉,而是Toast提示 结果10。说明程序已经被热修复,因为我们生成的dex文件中,将除数b改为了1,而这个正确的版本被安卓虚拟机预先加载了,所以不会执行我们程序中错误版本的代码。

相关文章

网友评论

    本文标题:一文带你手动实现最简单的Android热修复

    本文链接:https://www.haomeiwen.com/subject/hhlxzhtx.html