Android:这是一份全面&详细的 热修复 学习指南

时间:2022-06-19
本文章向大家介绍Android:这是一份全面&详细的 热修复 学习指南,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

前言

热补丁修复技术在Android 圈非常火,大量的热补丁方案开始大量涌现 本文将为你全面介绍热补丁的相关知识(原理、主流库使用),希望您会喜欢

目录

image.png

1. 简介

image.png

2. 储备知识

热补丁的原理主要基于: Android Dex分包方案 & Android的类加载机制(ClassLoader) 所以,在讲热补丁的原理前,先了解上述2个储备知识 2.1 Android Dex 分包方案 简介

image.png

示意图

image.png

2.2 Android 类加载机制(ClassLoader) 简介

image.png

加载流程说明

示意图

image.png

注:若2个Dex文件中有重复的类,当加载时,则优先加载排序较前的Dex文件的类

若所需加载类 = class3,则最终加载的是排序较前的Dex1文件中的class3

image.png

源码分析 由于 具体实现类 PathClassLoader、DexClassLoader都继承自BaseDexClassLoader类,所以此处主要讲解BaseDexClassLoader类中与类加载的相关方法findClass()

/**
   * 加载流程说明
   **/
   // 1. 传入需加载类的名字(classname)
   // 2. 通过Dex文件,寻找到所需类(findClass) 
       // a. 按顺序遍历ClassLoader的所有Dex文件,即 集合dexElements
       // b. 每遍历到1个Dex文件,则在该Dex文件中寻找所需加载的类
       // c. 若在该Dex文件找到该类,则返回;若找不到,则继续遍历下1个Dex文件
   // 3. 加载所需类

 /**
   * BaseDexClassLoader的findClass()源码分析
   **/
   @Override
   protected Class<?> findClass(String name) throws ClassNotFoundException {

       // 从pathList对象对象中寻找->>分析1
       Class clazz = pathList.findClass(name);

       if (clazz == null) {
           throw new ClassNotFoundException(name);
       }

       return clazz;
   }

 /**
   * 分析1:DexPathList的findClass()源码分析
   **/
   public Class findClass(String name) {
       // 1. 按顺序遍历ClassLoader的所有Dex文件,即 集合dexElements
       for (Element element : dexElements) {
           DexFile dex = element.dexFile;
           
           // 2. 每遍历到1个Dex文件,则在该Dex文件中寻找所需加载的类 ->>分析2
           if (dex != null) {
               Class clazz = dex.loadClassBinaryName(name, definingContext);
               // 3. 若在该Dex文件找到该类,则返回;若找不到,则继续遍历下1个Dex文件
               if (clazz != null) {
                   return clazz;
               }
           }
       }

       return null;
   }

 /**
   * 分析2:DexFile的loadClassBinaryName()源码分析
   **/
   public Class loadClassBinaryName(String name, ClassLoader loader) {
       return defineClass(name, loader, mCookie);
   }

 /**
   * 分析3:DexFile的defineClass()源码分析
   **/
   private native static Class defineClass(String name, ClassLoader loader, int cookie);
3. 热修复 原理

3.1 具体描述 把需修复、含Bug的类 独立打包到1个Dex文件中(记为:patch.dex) 将该 Dex文件 插入到ClassLoader中集合 dexElements的最前面 3.2 示意图

image.png

3.3 特别注意:CLASS_ISPREVERIFIED 标记 具体描述

image.png

解决方案具体描述

image.png

示意图

image.png

注:需完成上述步骤(防止类被打上 CLASS_ISPREVERIFIED 标记),再实现补丁

4. 热修复 开源库介绍

约在15年下半年开始,热补丁修复技术在 Android 圈非常火爆,热补丁方案开始大量涌现 下面,我将主要介绍当前主流的热修复开源库 4.1 主流的热修复 开源库

image.png

4.2 对比

image.png

5. 总结

本文主要讲解 Android中的热补丁相关知识 这些只是我个人的见解和经验,希望大家踊跃讨论,交流一下你们的宝贵经验,互相提高下!