专栏名称: Carson_Ho
走在产品路上的Android研究生
目录
相关文章推荐
开发者全社区  ·  大S的5亿遗产如何分配? ·  3 小时前  
开发者全社区  ·  和导师外出,他牵我手了... ·  昨天  
开发者全社区  ·  天王回乡的瓜 ·  2 天前  
开发者全社区  ·  程序员现在是个高危职业 ·  3 天前  
开发者全社区  ·  王J林在崩溃边缘 ·  3 天前  
51好读  ›  专栏  ›  Carson_Ho

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

Carson_Ho  · 掘金  · android  · 2019-02-28 01:09

正文

前言

  • 热补丁修复技术在 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 具体描述

  1. 把需修复、含 Bug 的类 独立打包到1个 Dex 文件中(记为: patch.dex
  2. 将该 Dex 文件 插入到 ClassLoader 中集合 dexElements 的最前面

3.2 示意图

示意图

3.3 特别注意:CLASS_ISPREVERIFIED 标记

  • 具体描述

示意图

  • 解决方案具体描述
    示意图

  • 示意图

示意图

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


4. 热修复 开源库介绍

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

4.1 主流的热修复 开源库

库名 作者 Github地址
Tinker 腾讯 微信团队 github.com/Tencent/tin…
Nuwa 腾讯 QQ空间团队 github.com/Tencent/tin…
Dexposed 阿里 手机淘宝团队 github.com/alibaba/dex…
AndFix 阿里 支付宝团队 github.com/alibaba/And…

4.2 对比

示意图







请到「今天看啥」查看全文


推荐文章
开发者全社区  ·  大S的5亿遗产如何分配?
3 小时前
开发者全社区  ·  和导师外出,他牵我手了...
昨天
开发者全社区  ·  天王回乡的瓜
2 天前
开发者全社区  ·  程序员现在是个高危职业
3 天前
开发者全社区  ·  王J林在崩溃边缘
3 天前