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

文章目录

一、修复包 Dex 文件准备

二、Odex 优化

三、Dex 文件拷贝

四、 源码资源





一、修复包 Dex 文件准备


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


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


修复后代码 :


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 文件 ;



进入到 " HotFix\app\build\intermediates\javac\debug\classes " 目录 , 这是生成 Class 字节码的目录 , HotFix 是 Android Studio 工程根目录 ;


删除 " HotFix\app\build\intermediates\javac\debug\classes\kim\hsl\hotfix " 目录中除了 HotFixTest.class 之外的其它字节码文件 ;



执行命令 :


Y:\001_DevelopTools\002_Android_SDK\build-tools\30.0.2\dx.bat


–dex --output


Y:\002_WorkSpace\001_AS\HotFix\app\build\intermediates\javac\debug\classes\update.dex


Y:\002_WorkSpace\001_AS\HotFix\app\build\intermediates\javac\debug\classes


Y:\001_DevelopTools\002_Android_SDK\build-tools\30.0.2\dx.bat --dex --output Y:\002_WorkSpace\001_AS\HotFix\app\build\intermediates\javac\debug\classes\update.dex

Y:\002_WorkSpace\001_AS\HotFix\app\build\intermediates\javac\debug\classes\


指令命令后 , 在 " Y:\002_WorkSpace\001_AS\HotFix\app\build\intermediates\javac\debug\classes\ " 目录下生成了需要更新的 " update.dex " 文件 ;

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



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



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






二、Odex 优化


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


Android 应用第一次安装比较慢 , 这是由于在 Android 5.0 5.05.0 之后的系统版本中 , 添加了 ART 虚拟机 , 5.0 5.05.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 文件拷贝到该目录中 ;


       

// 拷贝的目的文件目录
        // /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


上一篇:Chrome 浏览器降级后浏览网站不保留用户数据问题原因及解决方法


下一篇:数据库hash分库