美文网首页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