Android开发:热修复 Tinker 原理分析

文章讲述了如何在Android应用中使用MultiDex进行分包配置,创建SecondActivity作为异常处理入口,并利用FixDexUtils实现热修复功能,通过自定义DexClassLoader加载和修复私有目录中的dex文件,以解决10/0异常。热修复需重启应用才能生效,涉及类加载机制和设计模式的应用优化策略。
摘要由CSDN通过智能技术生成

public class BaseApplication extends MultiDexApplication {

@Override

public void onCreate() {

super.onCreate();

}

@Override

protected void attachBaseContext(Context base) {

super.attachBaseContext(base);

//安装分包配置

MultiDex.install(this);

}

}

2、分包配置

分包就创建了一个SecondActivity类做模拟异常和修复异常的入口,和一个Calculate模拟异常,做了10/0的操作,修复后为10/1

注:获取修复后的classes2.dex文件可以通过直接buildapk直接解压获取,或者用build-tools下的dx.bat执行命令获取

简单贴下SecondActivity,点击fix按钮后的代码:

private void update() {

//将下载的修复包,复制到私有目录,解压从.dex文件中取到对应的.class文件

//从sd卡取修复包

File sourceFile = new File(Environment.getExternalStorageDirectory(), Constants.DEX_NAME);

//目标文件

File targetFile = new File(getDir(Constants.DEX_DIR, Context.MODE_PRIVATE).getAbsolutePath() + File.separator + Constants.DEX_NAME);

if (targetFile.exists()) {

targetFile.delete();

Log.e(“update”,“删除原有dex文件(已使用的)”);

}

//将SD卡中的修复包copy到私有目录

FileUtils.copyFile(sourceFile,targetFile);

Log.e(“update”,“copy完成”);

FixDexUtils.loadDexFile(this);

}

3、FixModule

新建一个Module处理热修复的相关逻辑

只有五个文件,核心文件代码在FixDexUtils,其它是工具类,还有个定义了几个常量 看下FixDexUtils中的代码

public class FixDexUtils {

//修复文件可能有多个

private static HashSet loadedDex = new HashSet<>();

//不建议这么写,demo演示用

static {

loadedDex.clear();

}

public static void loadDexFile(Context context) {

//获取私有目录

File fileDir = context.getDir(Constants.DEX_DIR, Context.MODE_PRIVATE);

//遍历筛选私有目录中的.dex文件

File[] listFiles = fileDir.listFiles();

for (int i = 0; i < listFiles.length; i++) {

//文件名以.dex结尾,且不是主包.dex文件

if (listFiles[i].getName().endsWith(Constants.DEX_SUFFIX) && !“classes.dex”.equalsIgnoreCase(listFiles[i].getName())) {

loadedDex.add(listFiles[i]);

}

}

//创建自定义的类加载器

createDexClassLoader(context ,fileDir);

}

/**

  • @param context

  • @param fileDir

  • 创建自己的类加载器,加载私有目录的.dex文件,上面已经将修复包中的dex文件copy到私有目录

*/

private static void createDexClassLoader(Context context, File fileDir) {

//解压目录

String optimizedDir = fileDir.getAbsolutePath()+File.separator+“opt_dex”;

File fileOpt = new File(optimizedDir);

if (!fileOpt.exists()) {

fileOpt.mkdirs();

}

for (File dex : loadedDex) {

//创建自己的类加载器,临时的

DexClassLoader classLoader = new DexClassLoader(dex.getAbsolutePath(), optimizedDir, null, context.getClassLoader());

//有一个修复文件,就插装一次

hotFix(classLoader,context);

}

}

private static void hotFix(DexClassLoader classLoader, Context context) {

try {

//获取系统的PathClassLoader类加载器

PathClassLoader pathClassLoader = (PathClassLoader)context.getClassLoader();

//获取自己的dexElements数组

Object myElements = ReflectUtils.getDexElements(ReflectUtils.getPathList(classLoader));

//获取系统的dexElements数组

Object systemElements = ReflectUtils.getDexElements(ReflectUtils.getPathList(pathClassLoader));

//合并数组,并排序,生成一个新的数组

Object dexElements=ArrayUtils.combineArray(myElements,systemElements);

//通过反射获取系统的pathList属性

Object systemPathList = ReflectUtils.getPathList(pathClassLoader);

//通过反射,将合并后新的dexElements赋值给系统的pathList

ReflectUtils.setFieldValue(systemPathList,“dexElements”,dexElements);

}catch (Exception e){

e.printStackTrace();

}

}

}

主要做的工作就是:就是上面那张流程图,就是先通过遍历,解压等操作,获取到需要执行热修复的.dex文件集合,遍历该集合,对应每次创建一个临时的DexClassLoader,然后执行修复步骤,细分就是六步:

最终实现效果如图(Demo使用手机是华为8.0的手机):

注:这里为了效果图更直观,已经重启过一次app

注: 这种方式实现的热修复必须要重启App才可以实现修复,这一点也是类加载机制决定的,如下图中修复之后再次打开加载执行修复后的classes.dex文件是在BaseApplication中调用了修复方法

最后

==============================================================

在这里我也分享一份由几位大佬一起收录整理的Android学习PDF+架构视频+面试文档+源码笔记高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料

如果你有需要的话,可以 点这领取

喜欢本文的话,不妨顺手给我点个小赞、评论区留言或者转发支持一下呗~



自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后说一下我的学习路线

其实很简单就下面这张图,含概了Android所有需要学的知识点,一共8大板块:

  1. 架构师筑基必备技能
  2. Android框架体系架构(高级UI+FrameWork源码)
  3. 360°Androidapp全方位性能调优
  4. 设计思想解读开源框架
  5. NDK模块开发
  6. 移动架构师专题项目实战环节
  7. 移动架构师不可不学习微信小程序
  8. 混合开发的flutter

Android学习的资料

我呢,把上面八大板块的分支都系统的做了一份学习系统的资料和视频,大概就下面这些,我就不全部写出来了,不然太长了影响大家的阅读。

330页PDF Android学习核心笔记(内含上面8大板块)

Android学习的系统对应视频

总结

我希望通过我自己的学习方法来帮助大家去提升技术:

  • 1、多看书、看源码和做项目,平时多种总结

  • 2、不能停留在一些基本api的使用上,应该往更深层次的方向去研究,比如activity、view的内部运行机制,比如Android内存优化,比如aidl,比如JNI等,并不仅仅停留在会用,而要通过阅读源码,理解其实现原理

  • 3、同时对架构是有一定要求的,架构是抽象的,但是设计模式是具体的,所以一定要加强下设计模式的学习

  • 4、android的方向也很多,高级UI,移动架构师,数据结构与算法和音视频FFMpeg解码,如果你对其中一项比较感兴趣,就大胆的进阶吧!

希望大家多多点赞,转发,评论加关注,你们的支持就是我继续下去的动力!加油!

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

w的内部运行机制,比如Android内存优化,比如aidl,比如JNI等,并不仅仅停留在会用,而要通过阅读源码,理解其实现原理

  • 3、同时对架构是有一定要求的,架构是抽象的,但是设计模式是具体的,所以一定要加强下设计模式的学习

  • 4、android的方向也很多,高级UI,移动架构师,数据结构与算法和音视频FFMpeg解码,如果你对其中一项比较感兴趣,就大胆的进阶吧!

希望大家多多点赞,转发,评论加关注,你们的支持就是我继续下去的动力!加油!

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 22
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值