【Android 热修复】热修复原理 ( 修复包 Dex 文件准备 | Dex 优化为 Odex | Dex 文件拷贝 | 源码资源 )

2023-03-29 09:40:29 浏览数 (1)

文章目录

  • 一、修复包 Dex 文件准备
  • 二、Odex 优化
  • 三、Dex 文件拷贝
  • 四、 源码资源

一、修复包 Dex 文件准备


异常代码 : 故意写一个异常代码 , 并执行该代码 , 肯定会崩溃 ;

代码语言:javascript复制
package kim.hsl.hotfix;

import android.util.Log;

public class HotFixTest {
    public void test(){
        if (true) {
            // 直接抛出异常
            throw new RuntimeException();
        }
        Log.i("HotFixTest", "HotFixTest 执行成功");
    }
}

修复后代码 :

代码语言:javascript复制
package kim.hsl.hotfix;

import android.util.Log;

public class HotFixTest {
    public void test(){
        if (false) {
            // 直接抛出异常
            throw new RuntimeException();
        }
        Log.i("HotFixTest", "HotFixTest 执行成功");
    }
}

参考 【Android 热修复】热修复原理 ( 热修复框架简介 | 将 Java 字节码文件打包到 Dex 文件 ) 二、 将 Java 字节码文件打包到 Dex 文件 章节流程 , 将更新后的 kim.hsl.hotfix.HotFixTest 类打包成 dex 文件 ;

进入到 " HotFixappbuildintermediatesjavacdebugclasses " 目录 , 这是生成 Class 字节码的目录 , HotFix 是 Android Studio 工程根目录 ;

删除 " HotFixappbuildintermediatesjavacdebugclasseskimhslhotfix " 目录中除了 HotFixTest.class 之外的其它字节码文件 ;

执行命令 :

Y:01_DevelopTools02_Android_SDKbuild-tools30.0.2dx.bat

–dex --output

Y:02_WorkSpace01_ASHotFixappbuildintermediatesjavacdebugclassesupdate.dex

Y:02_WorkSpace01_ASHotFixappbuildintermediatesjavacdebugclasses

代码语言:javascript复制
Y:01_DevelopTools02_Android_SDKbuild-tools30.0.2dx.bat --dex --output Y:02_WorkSpace01_ASHotFixappbuildintermediatesjavacdebugclassesupdate.dex Y:02_WorkSpace01_ASHotFixappbuildintermediatesjavacdebugclasses

指令命令后 , 在 " Y:02_WorkSpace01_ASHotFixappbuildintermediatesjavacdebugclasses " 目录下生成了需要更新的 " update.dex " 文件 ;

( 可以参考 【Android 安全】DEX 加密 ( Java 工具开发 | 生成 dex 文件 | Java 命令行执行 ) 博客内容)

将编译好的 update.dex 放置到 SD 卡根目录中 , 下一步开始使用该 dex 文件进行热修复 ;

二、Odex 优化


修复包就是一个 Dex 文件 , 热修复框架虽然给生成了一个 .patch 文件 , 其本质还是将 Dex 文件封装到了 .patch 文件中 ;

Android 应用第一次安装比较慢 , 这是由于在 Android 5.0 之后的系统版本中 , 添加了 ART 虚拟机 , 5.0 之前的系统版本安装速度很快 ;

老版本的手机安装 APK 应用 , 只需要几秒钟时间 ; 新版本的手机 , 安装 APK 文件会慢很多 ;

新版手机系统安装时 , 对 Dex 文件进行了优化 , 将 Dex 文件转为 Odex 文件 , 系统查找类时 , 加载 Dex 速度很慢 , 将 Dex 优化为 Odex , 从 Odex 中取 Class 字节码速度要更快一些 ;

三、Dex 文件拷贝


应用运行时不会从 APK 压缩文件中加载 Class 字节码文件 , APK 文件安装在 Android 手机中时 , 将 APK 中的压缩文件拷贝到 /data/user/包名/app_odex 目录下 ;

首先获取 /data/user/包名/app_odex 目录 , 使用 Context.getDir 目录时 , 会自动在字符串前添加 “app_” 前缀 , 先将 Dex 文件拷贝到该目录中 ;

代码语言:javascript复制
        // 拷贝的目的文件目录
        // /data/user/0/kim.hsl.hotfix/app_odex
        File targetDir = this.getDir("odex", Context.MODE_PRIVATE);
        // 拷贝的目的文件名称
        String targetName = "update.dex";

        // 准备目的文件, 将 Dex 文件从 SDK 卡拷贝到此文件中
        String filePath = new File(targetDir, targetName).getAbsolutePath();
        File file = new File(filePath);
        if (file.exists()) {
            file.delete();
        }

        // 准备输入流, 读取 SD 卡文件
        InputStream is = null;
        // 准备输出流, 输出到目的文件
        FileOutputStream os = null;

        try {
            // 读取 SD 卡跟目录的 update.dex 文件
            is = new FileInputStream(new File(Environment.getExternalStorageDirectory(), targetName));
            // 输出到目标文件
            os = new FileOutputStream(filePath);
            int len = 0;
            byte[] buffer = new byte[1024];
            while ((len = is.read(buffer)) != -1) {
                os.write(buffer, 0, len);
            }

            // 进行后续操作

        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 关闭 IO 流
            try {
                os.close();
                is.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }

四、 源码资源


源码资源 :

  • GitHub 地址 : https://github.com/han1202012/HotFix
  • CSDN 源码快照 : https://download.csdn.net/download/han1202012/16641987

0 人点赞