正文
前言
-
热补丁修复技术在
Android
圈非常火,大量的热补丁方案开始大量涌现
-
本文将为你全面介绍热补丁的相关知识(原理、主流库使用),希望您会喜欢
目录
1. 简介
2. 储备知识
-
热补丁的原理主要基于:
Android Dex
分包方案 &
Android
的类加载机制
(ClassLoader)
-
所以,在讲热补丁的原理前,先了解上述2个储备知识
2.1 Android Dex 分包方案
2.2 Android 类加载机制(ClassLoader)
-
简介
-
加载流程说明
-
示意图
注:若2个
Dex
文件中有重复的类,当加载时,则优先加载排序较前的
Dex
文件的类
若所需加载类 =
class3
,则最终加载的是排序较前的
Dex1
文件中的
class3
-
源码分析
由于 具体实现类
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 示意图
3.3 特别注意:CLASS_ISPREVERIFIED 标记
-
解决方案具体描述
-
示意图
注:需完成上述步骤(防止类被打上
CLASS_ISPREVERIFIED
标记),再实现补丁
4. 热修复 开源库介绍
-
约在15年下半年开始,热补丁修复技术在
Android
圈非常火爆,热补丁方案开始大量涌现
-
下面,我将主要介绍当前主流的热修复开源库
4.1 主流的热修复 开源库
4.2 对比