原文 http://blog.votzone.com/2018/05/12/apk-merge.html

之前接手过一个sdk的开发工作,在开发过程中有一个很重要的点就是尽量使用代码来创建控件,资源文件最好放到assets目录下,如果必须使用res资源,需要通过 getResources().getIdentifier("activity_splash","layout", getPackageName()) 这种方式来获取资源id,而不能直接通过R文件获取。

今天就来研究一下这个问题。

一、lib项目中r文件中资源唯一标志为static变量

一般的app项目中自动生成的R文件为常量,而在library项目中为变量。根据Android官方文档,在android 14 之后添加的这一特性,之前编译后的lib项目中是常量,之后的为static 变量。 目的是为了在资源冲突时能够修改资源唯一值。 如图在library项目中,自动生成的R文件如下

有一个需要关注的点是R文件是java 代码,在build时会生成.class文件并添加到dex中。 因为R文件中的常量值仅仅受编译器控制,在lib发布之后添加到jar包中的.class并不会受到当前编译器的影响。而通常lib发布之后要给第三方使用。

看一下反编译后的apk

反编译后查看mylib 下对应的R文件smali代码,可见其值又被设置为final(常量)了。 因此我们可以知道编译器在生成apk时虽然没有lib的java代码可以重置和修改,但是在将jar转化为dex时可以转换为常量。 转化为常量后并不影响其使用, 如图在lib中使用layout资源的反编译代码

我们可以看出使用静态常量可以在生成apk时动态修改其指定的值。

如果是常量,编译后v0的值将直接使用常量值,这样修改R.class文件中的值将没有意义。例如在app中反编译后代码如下:

二、从反编译后代结构中查看资源对应问题

可以看到,反编译后的代码与我们写的java代码基本是一一对应的,那么问题来了,Android是怎么通过一个id值来找到需要的资源呢?通过分析Android源码我们当然可以找出过程,但是分析apk反编译后的结构可以给我们更直接的思路。

在res/values/文件夹下有个public.xml文件,其中每行有三对值,分别为typenameid 通过分析我们可以直接得出结论: public.xml中的对应关系直接关系到哪个id找那个资源。

三、添加资源

Unity3d和cocos2d引擎有自己一套方式来添加资源id,假如我们自己搞一个游戏,又想为其添加一些java代码和资源该怎么操作?

根据之前的分析我们可以设计如下测试方案

  1. 假定我们要添加代码的apk为targetapp,我们编写一个叫mergelib的apk,然后将mergelib中资源和代码添加到targetapp中。
  2. 在编写mergelib时,使用 getResources().getIdentifier("activity_splash","layout", getPackageName()) 的方式获取资源id;
  3. 通过apktool反编译代码,将资源文件复制到要加入的目标apk中;
  4. 将mergelib的public.xml文件中需要的资源项添加到targetapp中;
  5. 编译并签名测试。

根据如上实验我们可以确定这样的操作是可行的。测试流程如下:

需要的资源:

  • targetapp- 目标app, 在这里代表游戏
  • mergelib- 要将其中包含资源的代码合并进去
    mergelib 中对资源通过getIdentifier()的方式使用: 例如设置启动页Activity中的ContentView的设置
int id = getResources().getIdentifier("activity_splash","layout", getPackageName());    
setContentView(id);

我们的目标:为targetapp添加一个启动页,启动页代码在mergelib中编写。

执行流程

  1. 修改targetapp中AndroidManifest.xml文件 
    1) 将SplashActivity 的声明添加进去 
    2) 修改启动Activity

  2. 复制需要的代码进入targetapp: 
    复制mergelib中SplashActivity的代码并修改启动MainActivity的启动代码 如图

  3. 复制资源
    1) 将res/anim 下alpha.xml复制到target
    2) 将res/layout 下 spalsh.xml复制到target 下

  4. 修改ids 
    将 res/values/ids.xml 中多出来的行复制到 对应ids.xml文件中

    如图, 本例中仅有一个id 即ImageView的id, 因此将该行复制到targetapp 中对应的ids.xml文件中即可 位置不重要

  5. 修改public.xml文件 mergelib Splash中用到了 animlayoutid, 并且间接用到了 mipmap,因此这些对应的值都需要添加到targetapp。观察public.xml的文件结构,可以发现如下特点: 
    1) 所有同类型(type相同)的id连续 
    2) 同类型的id 前4字节相同, 如下图 anim 的前四字节0x7f01 与 attr 不同

    3) 所有id唯一 
    根据以上三个特点,我们将多出来的id添加到target为了保证唯一且方便修改,我们做了如下替换(右侧为target)

  6. 一切就绪,编译并安装

四、工具化处理public.xml的替换过程

上述手动测试仅仅只有一个资源id的情况,假如我们加入了一个support包或者其他一些包含资源的包,那么资源数量将会增加到几百个,这样的话手动添加肯定是不行的,我们需要一个脚本工具来实现。

脚本接收两个public.xml格式的文本,并输出一个合并版本。 其中targetapp中的id值是不可变得,在遇到id冲突时,我们改变mergelib的public.xml。

1) 复制mergelib的public文件为mergeid.xml 

2) 复制target app 的public 并命名为oriid.xml

3) 将mergeid.xml 和oriid.xml 放到RIDreset.py同目录下, 运行py脚本

4) 出现 oriid.xml_ 文件, 即生成的合并文件

案例及脚本

https://github.com/votzone/DroidCode/tree/master/VotAndroid/Mergeapp

直接在apk中添加资源的研究的更多相关文章

  1. 使用NetBeans生成jar包,并在jar包中添加资源

    在NetBeans中,执行Clean and Build便可得到jar文件 若要在jar中添加资源,先用压缩软件打开jar,然后将资源拖进当前归档文件即可 使用Class.getResource(St ...

  2. apk中添加第三方so文件

    如果你是把so放在libs/armeabi/下,eclipse中so会自动打包进去,然后使用System.load("data/data/xxx.xxx.xxx/lib/xx.so" ...

  3. QT工程中添加资源(简单明了)

    1. 在工程文件下右击添加新文件 2. 在QT目录下选择QT Resource File 3. 填写资源名称 4. 点击完成就可以看到自己建立的资源了 5. 点击右键添加现有文件,找到自己要添加的资源 ...

  4. Android插件化(三)载入插件apk中的Resource资源

    Android载入插件apk中的Resource资源 简单介绍 怎样载入未安装apk中的资源文件呢?我们从android.content.res.AssetManager.java的源代码中发现,它有 ...

  5. 完美解决SpringMVC中静态资源无法找到(No mapping found for HTTP request with URI)问题

    https://blog.csdn.net/kingmax54212008/article/details/79330308 今天遇到一个比较新奇的问题,但是也应该是使用spring MVC框架时由于 ...

  6. 【初码干货】使用阿里云对Web开发中的资源文件进行CDN加速的深入研究和实践

    提示:阅读本文需提前了解的相关知识 1.阿里云(https://www.aliyun.com) 2.阿里云CDN(https://www.aliyun.com/product/cdn) 3.阿里云OS ...

  7. 34.如何获取app(apk和ipa)中的资源

    移动互联网中,主要的两个平台是android和ios,android上文件的安装包是后缀名为apk的文件,ios上文件的安装包是后缀名为ipa的文件,在本文分析一下这两种文件的特点,以及如何用程序去解 ...

  8. WPF 添加 Resources Dictionary 资源 一般类库项目中无法添加资源文件(ResourceDictionary)

    在文件夹或者项目右键-> Add(添加),会弹出可以快捷添加的资源,但是你会发现没有 ResourceDictionary资源可以选择. 解决此问题方法: 第一步:工程->右键->U ...

  9. 【转】QT中添加的资源文件qrc时的路径问题小结

    @2019-06-13 [小记] QT中添加的资源文件qrc时的路径问题小结

随机推荐

  1. js中常用的算法排序

    在工作中都会经常用到的一些基础算法,可以很快解决问题.这些都是在工作中总结的,希望可以帮助到大家. 一.数组乱序 arr.sort(function randomsort(a, b) { return ...

  2. MyEclipse 2017/2018 安装与破解 图文教程

    SSM 框架-02-MyEclipse 2017/2018 安装与破解 现在在学J2EE,然后使用的工具就是 MyEclipse,现在就抛弃 Eclipse 了,我就不多说它俩的区别了,但是 MyEc ...

  3. 结对编程项目总结(core2组)

    结对编程项目总结(core2组) 作业---四则运算(Core 第二组)   ----by 吴雪晴 PB16061514 齐天杨 PB16060706 一.项目简介 项目的任务为制作一个给(貌似是?) ...

  4. 三年Linux运维工作总结教训

    Linux运维一定要知道的六类好习惯和23个教训,避免入坑! 从事运维三年半,遇到过各式各样的问题,数据丢失,网站挂马,误删数据库文件,黑客攻击等各类问题. 今天简单整理一下,分享给各位小伙伴. 一. ...

  5. 打通版微社区(5):部署DZ3.2

    参考官方帖子http://www.discuz.net/thread-3258186-1-1.html  这是第三方的帖子http://www.discuz.net/thread-3199850-1- ...

  6. SDN第一次作业——关于网络编程

    你会选择作网络编程方向的程序员吗?为什么? 我觉得我应该不会选择成为专门从事网络编程的程序员,因为我的从事计算机的理想一直是希望开发出自己的一款软件和游戏,所以应该是比较偏向软工方向的,也报了栋哥的软 ...

  7. [T-ARA][한겨울의 숨바꼭질/숨바꼭질][深冬的捉迷藏/捉迷藏]

    歌词来源: 深冬的捉迷藏:http://music.163.com/#/song?id=28111190 捉迷藏:http://music.163.com/#/song?id=28111189 作曲 ...

  8. Mysql中函数和存储过程的区别

    Mysql中函数和存储过程的区别 存储过程: 1.       可以写sql语句 2.       inout,out构造返回值 3.       调用:call:存储过程名称 4.       可以 ...

  9. luogu P2015 二叉苹果树

    嘟嘟嘟 这应该算一道树形背包吧,虽然我还是分不太清树形背包和树形dp的区别…… 首先dp[i][u][j] 表示在u的前 i 棵子树中,留了 j 条树枝时最大的苹果数量,而且根据题目描述,这些留下的树 ...

  10. Hadoop学习之路(二十二)MapReduce的输入和输出

    MapReduce的输入 作为一个会编写MR程序的人来说,知道map方法的参数是默认的数据读取组件读取到的一行数据 1.是谁在读取? 是谁在调用这个map方法? 查看源码Mapper.java知道是r ...