安卓热修系列-插件资源冲突解决方案

作者:37手游移动客户端团队

背景

在做插件化过程中,宿主需要用到插件的资源,涉及到加载插件的资源;

因为插件是以apk的方式存在的,所以插件的ID和宿主的ID可能导致重复;

为了解决这个问题,需要把插件的资源ID重新排一遍,才给宿主加载和使用。

资源加载

Android项目中的资源常规是通过R文件来索引的。打包的时候aapt将工程中的资源名与id在R文件中映射起来。使用资源时是通过Resources获取的,如:

resources.getDrawable(xxxid)

那么怎么加载插件中的资源呢?先看Resources的构造方法:

public Resources(AssetManager assetManager, DisplayMetrics metrics, Configuration config) {
}

Resources的初始化依赖:AssetManager,DisplayMetrics和Configuration。

在AssetManager中有一个方法

public int addAssetPath(String path)

该方法可用于AssetManager增加外部资源,但是AssetManager中该方法不对外公开,需要反射调用,代码大概如下:

val method = AssetManager::class.java.getDeclaredMethod("addAssetPath", String::class.java)
method.isAccessible = true
method.invoke(hostAssetManager, apk)

资源冲突方案

经过上述的处理后,插件的资源和宿主的资源是混合再一起的,因为所有的资源索引不做处理时都是以0x7faabbbb格式按顺序生成的。所以会存在插件资源ID和宿主资源ID冲突的情况(注意:资源索引共四个字节32位,第一个字节代表PackgeID,第二个字节代表TypeID,后两个字节代表资源值)

怎么解决冲突呢?有几种方案:

方案一:资源隔离

资源隔离就是宿主和插件使用不同的Resources对象,这样使用的资源文件不同,不存在冲突。代码调整如下:

//通过反射生成一个新的AssetManager实例
val am = AssetManager.class.newInstance()
//添加资源
val method = AssetManager::class.java.getDeclaredMethod("addAssetPath", String::class.java)
method.isAccessible = true
method.invoke(am, apk)
//生成新的Resources对象
val pluginResources = Resources(am, hostResources.displayMetrics, hostResources.configuration)

插件使用这个PluginResources。但是这里会有一些缺点:

1、宿主和插件的资源无法共享,使用起来不太方便

2、PluginResouces中不包含系统资源,在某些用到系统资源的场景会报错。如:加载前端页面,使用select标签时会使用到系统的一些资源

3、某些业务场景下,插件的Resources和宿主的Resources需要用同一个。如:游戏发行sdk,sdk相关界面非activity,使用到的Resources是游戏研发方activity(在宿主中)。

方案二:修改资源id

修改资源id,当前常用的方案有:

方案1:Android的资源id是aapt生成的,修改aapt,让插件中的资源id不从0x7f开始,比如从0x6f开始

方案2:生成插件apk后,修改插件apk中resources.arsc文件。resources.arsc文件是有固定格式的文件

可通过解析该文件修改资源索引值。但是存在一定缺陷:

1、修改难度较大,需要了解resources.arsc格式

2、只修改resources.arsc是不够的,因为代码中使用的是R类,还同时需要修改代码中的值;

方案3:通过反编译修改插件中的public.xml文件中的索引值,以及修改smali中的R类值

public.xml这个文件是哪来的?

该文件是apktool在反编译apk时,根据apk包中的resources.arsc文件生成。 没看过resource.arsc? (自己拖个apk到IDE看吧)

public.xml有什么作用

publc.xml是aapt在打包资源时用来固定资源id的,如果资源在public.xml中有对应的id了,那么打包资源时就用已经有的id。

public.xml中的id的格式

共四个字节32位,第一个字节代表PackgeID,第二个字节代表TypeID,后两个字节代表资源值

通常系统资源PackageID是01,而我们自己的资源PackageID是7f

TypeID,比如attr为01,string为02。但是并不固定,并不一定attr就是01。但是在public.xml中,同类型的该字节一定是一样的,否则回编译会失败。

R类

R类这里有个知识点,library模块中生成的R类中的成员的值不是常量,不带final。app模块生成的R类的值是常量值。而常量值在java编译时会被优化,最终代码中输出的就是常量值,而不是R.id.xxx这样。而library的因为是变量,不会被优化,代码中会保留R.id.xxx

R类和public.xml的关系

从本质上讲,其实并没有啥关系。但是由于在代码中我们会使用R.id去查找资源,这就关联上了。如果都用getIdentifier的方式先获取id,那把R类删了也没事。

public.xml打包后对应的就是resources.arsc中的值,而资源值生成Java类,这个类就是R类。也就是说平时使用R类,就是用里面的索引值去到resources.arsc中找到对应资源位置,再去加载。

修改步骤如下:

1、反编译插件apk

2、修改public.xml,将0x7faabbbb中的aa的值调高,比如0x7f010001→0x7f510001,因为第二个字节是TypeID,是按顺序增加的,而这个type实际上是没有多少的。修改成比较大的值后就不会冲突了。 (具体修改可参考提供代码中的PublicXmlBean类实现)

3、修改R$x.smali的值,反编译后R类就是一些smali代码,扫描smali代码,将其中的值也都按同样的规则增大。这样R类和resources.arsc依旧是对应的,代码中的R类正常。但是这里需要注意的是application模块的R类由于是final常量值,会被处理成常量。这种情况下修改R类其实不会有作用。而library的情况则可以。因此需要将涉及到R的代码及资源文件下沉为library模块依赖。

4、回编译,得到修改资源索引后的插件

源码工程简介

概要

程序入口

插件生成小tips

插件源码工程中的插件apk结构(也就是plugin.apk):

实际代码下沉为resmodule作为library模块,app模块中并无实际功能,仅仅作为application模块引用library,这样就可以输出我们的插件plugin.apk。 这样做的目的上面有说,其实就是避免我们的代码等在app模块,被弄成常量了。

效果

用工具修改前的R.layout.activity_main的值为2131296284,转换为16进制为0x7f09001c

用工具修改后再次查看R.layout.activity_main的值为2136539164,即0x7f59001c

总结

本文介绍了插件资源加载的多种方案,可以按照自己的使用场景以及团队技术能力选择合适的方案

Android 学习笔录

Android 性能优化篇:https://qr18.cn/FVlo89
Android 车载篇:https://qr18.cn/F05ZCM
Android 逆向安全学习笔记:https://qr18.cn/CQ5TcL
Android Framework底层原理篇:https://qr18.cn/AQpN4J
Android 音视频篇:https://qr18.cn/Ei3VPD
Jetpack全家桶篇(内含Compose):https://qr18.cn/A0gajp
Kotlin 篇:https://qr18.cn/CdjtAF
Gradle 篇:https://qr18.cn/DzrmMB
OkHttp 源码解析笔记:https://qr18.cn/Cw0pBD
Flutter 篇:https://qr18.cn/DIvKma
Android 八大知识体:https://qr18.cn/CyxarU
Android 核心笔记:https://qr21.cn/CaZQLo
Android 往年面试题锦:https://qr18.cn/CKV8OZ
2023年最新Android 面试题集https://qr18.cn/CgxrRy
Android 车载开发岗位面试习题:https://qr18.cn/FTlyCJ
音视频面试题锦:https://qr18.cn/AcV6Ap

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/8323.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【云原生】Docker部署/容器加速器(最新版)

目录 初时Docker和部署 1.什么是Docker 2.容器和虚拟化的区别 3.部署Docker 1.卸载历史版本 2.设置存储库 3.安装Docker最新引擎 4.安装Docker特定安装引擎 1.先查看当前docker-ce都有那些版本 2.替换为所需版本,然后运行以下命令 要安装的命令: 5.启动D…

攻防世界_web

robots 题目描述是这样的,虽然这是一道基础题,但我确实还没有了解过robots协议 第一次知道是被御剑给扫描出来的后台文件 这次直接访问看看 初级题就是初级题,访问后得到提示,如果没做过我估计还不知道该咋整,这也是一…

Linux服务器丢包故障的解决思路及引申的TCP/IP协议栈理论

Linux服务器丢包故障的解决思路及引申的TCP/IP协议栈理论 我们使用Linux作为服务器操作系统时,为了达到高并发处理能力,充分利用机器性能,经常会进行一些内核参数的调整优化,但不合理的调整常常也会引起意想不到的其他问题&#x…

ML@集成学习@摘要

文章目录 集成学习refs摘要Note准确性和多样性 集成学习方法分类BoostingAdaBoost伪代码Adaboost小结 补充补充1补充2 BaggingBagging伪代码 特点算法效率直接应用于多分类 自助采样和包外估计随机森林 Stackingsklearn中的Stacking🎈构造初级学习器构造次级学习器 …

Spring的数据访问哲学

目录 设计思路 了解Spring的数据访问异常体系 数据访问模板化 设计思路 Spring的目标之一就是允许我们在开发应用程序时,能够遵循面向对象(OO)原则中的“针对接口编程”Spring对数据访问的支持也不例外像很多应用程序一样,Spittr应用需要从某种类型的…

​山东大学高校专区入驻飞桨AI Studio,优质教育资源等你来学!

近日,山东大学高校专区在飞桨人工智能学习与实训社区 AI Studio 上线,双方将携手搭建人工智能教学实训平台专区,汇集优质教学实训资源,校企共同培育复合型 AI 人才,为国家输送高质量人才,促进国家智能化进程…

保姆级python环境配置(anaconda+pycharm+cuda+cudnn+pytorch)

文章目录 前言一、如何下载anaconda1、下载网址2、版本选择3、下载流程4、注意事项 二、如何下载pycharm1、下载网址2、下载流程 三、更新NVIDIA驱动1、下载网址2、选择相应配置进行下载 四、如何下载cuda1、查看可安装的cuda版本号2、下载网址3、下载流程4、注意事项 五、如何…

2023年上海市浦东新区网络安全管理员决赛理论题样题

目录 一、判断题 二、单选题 三、多选题 一、判断题 1.等保1.0至等保2.0从信息系统拓展为网络和信息系统。 正确 (1)保护对象改变 等保1.0保护的对象是信息系统,等保2.0增加为网络和信息系统,增加了云计算、大数据、工业控制系统、物联网、移动物联技术、网络基础…

手工测试没有前途,自动化测试会取代手工测试?

在测试行业,一个一直被讨论的问题就是:手工测试没有前途,自动化测试会取代手工测试? 首先说结论:自动化测试不会取代手工测试,这完全是两个维度的事情。为什么不会呢?我们需要从本源上说起。 什…

科技项目验收测试报告包括哪些内容?

科技项目验收测试报告是评估科技项目质量和可靠性的重要文件。通过全面的测试和评估,可以确保项目的质量,提高用户满意度,降低项目风险。 一、科技项目验收测试报告的内容 1. 项目概述:介绍项目的背景、目标和范围,…

MySQL学习基础篇(七)---单行函数

MySQL学习基础篇(七)—单行函数 1 什么是函数 函数在计算机语言的使用中贯穿始终,函数的作用是什么呢?它可以把我们经常使用的代码封装起来,需要的时候直接调用即可。这样既 提高了代码效率 ,又 提高了可维护性 。在 SQL 中我们…

三种Linux内核代码在线阅读工具

记录一下 1 . 可在线阅读uboot,kernel,busybox(rootfs),可搜索字符串,函数跳 https://lxr.missinglinkelectronics.com/ 界面如下: 2. 显示界面跟代码编辑器很像,同样可以函数跳转 https://elixir.boot…