其他
向工程腐化开炮|资源治理
系列文章回顾《向工程腐化开炮 | proguard治理》《向工程腐化开炮 | manifest治理》《向工程腐化开炮:java代码治理》。本文为系列文章第四篇,聚焦于Android 资源,这一细分领域。对工程腐化,直接开炮!
准确的说,本文主角是Android资源,而java资源归属到java代码治理范畴,并在「向工程腐化开炮:java代码治理」一文中给出了应对方案。
Android资源从定义和使用方式来看,可以分为Resource和Asset两个大类。前者提供受控的结构化访问方式,每个资源均有唯一id标识,以及多种配置限定符来支持多语言、多设备、多特性等能力;后者提供原始且相对自由的目录和文件访问。Resource类型是绝大部分资源使用场景下的最佳选择,本文主要聚焦的即是这种类型资源,对冲突、无用、缺失类引用、硬编码文本,这几种腐化情况,开展工具研发,以及治理实践。
R.<type>.<name>
形式在java代码中引用,其中一些还可以通过@<type>/<name>
形式在manifest和资源中引用。对上述分类中「是否独立文件」、「是否位于resources.arsc」两个维度进行一些解读:是否独立文件。一个资源如果对应一个完整的独立文件,这种属于File-Base Resource,在最终apk的res目录下也会存在一份对应文件;否则,属于Value-Base Resource,在apk中没有独立文件与之对应,其值(如果有)存储在resources.arsc中。其中color类型比较特殊,单一color资源是Value-Base,但是颜色状态列表(ColorStateList)属于File-Base。此外,是否独立文件,是从资源编译后这一视角来看的,在定义资源时,Android提供了一种内嵌xml资源的形式,可以把多个独立文件类型资源,写在一个xml文件中,在此不展开讨论; 是否位于resources.arsc。绝大部分资源,在 R$<type>
类中field的取值,都是0x7fxxxxxx,并且在resources.arsc中都有一条对应记录。对于File-Base资源,记录值是file的相对路径,对于Value-Base资源,记录值就是资源值本身。需要注意的是,styleable类型资源比较特殊,仅存在于R$styleable
类中,其field取值也并不是0x7fxxxxxxx格式,而是整型或整型数组,并且在resources.arsc中并不存在。
# 资源定义于 res/value/attrs.xml
<resources>
<declare-styleable name="DeclareStyleable1" >
<attr name="attr_enum" format="enum">
<enum name="attrEnum1" value="1"/>
<enum name="attrEnum2" value="2"/>
</attr>
<attr name="attr_integer" format="integer"/>
<attr name="android:padding" format="dimension"/>
</declare-styleable>
</resources>
# R.java文件中,生成以下代码
public static final class id {
public static final int attrEnum1=0x7f060000;
public static final int attrEnum2=0x7f060001;
}
public static final class attr {
public static final int attr_enum=0x7f020000;
public static final int attr_integer=0x7f020001;
}
public static final class styleable {
public static final int[] DeclareStyleable1 = {0x010100d5, 0x7f020000, 0x7f020001};
public static final int DeclareStyleable1_android_padding=0;
public static final int DeclareStyleable1_attr_enum=1;
public static final int DeclareStyleable1_attr_integer=2;
}
# resources.arsc中,生成记录
type | id | name | value
id 0x7f060000 attrEnum1 None
id 0x7f060001 attrEnum2 None
attr 0x7f020000 attr_enum 1,2
attr 0x7f020001 attr_integer 0
一个attr,name使用android:xxxx,在R.java和resources.arsc中不会生成对应内容,因此在语意可复用时,使用系统提供的attr,可以节省一点包大小空间; 如果多个styleable或者style,定义了同名attr,实际只会生成一个attr资源,相当于提高了复用度; attrEnum1、attrEnum2这种id类型资源,如果其它类型资源(例如layout)中也有同名定义,那么实际也只会生成一个id资源,同样也提高了复用度。
AndroidManifest.xml文件。其中对资源的引用,会替换为对应资源id,并编译为二进制格式,最终会被打包到apk中。 resources.arsc文件。资源符号(索引)表,记录所有资源id与各配置下的资源值,最终也会被打包到apk中。 处理(编译)后的资源文件集合。所有需要编译的独立资源文件(例如layout),均会编译为二进制格式,和不需要编译的资源文件一起,最终被打包到apk中。 R.java文件。记录资源类型/名称,与id值的对应关系,用于在java代码中直接引用。每个模块(subproject、flat aar、external aar)都会生成对应的package.R.java文件,最终和其它所有java源文件一起,共同进行javac编译。 资源对应keep规则文件。主要包括layout中view节点对应java类,onClick属性值对应java方法,以及manifest中四大组件对应java类。这些keep规则,会与其它自定义keep规则一起,用于后续的proguard处理。
如果通过 Resources.getIdentifier
动态引用资源时,名称参数完全是一个变量,那么会导致相关资源被误删;如果java代码常量池中,几乎包含所有单个字符,例如a-z,1-9,那么所有资源均会被认为有引用,导致不会裁剪任何一个资源(优酷就是如此)。
1被忽视的一员 - id类型资源
# styleable类型资源,定义于 res/value/attrs.xml
<resources>
<declare-styleable name="DeclareStyleable1" >
<attr name="attr_enum" format="enum">
<enum name="attrEnum1" value="1"/>
<enum name="attrEnum2" value="2"/>
</attr>
<attr name="attr_integer" format="integer"/>
<attr name="android:padding" format="dimension"/>
</declare-styleable>
</resources>
# layout类型资源,定义于 res/layout/main.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/main_textview"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textColor="@color/purple_200"
android:text="Hello World!"/>
</LinearLayout>
2资源与java代码的桥梁 - R类
R.<type>.<name>
的引用,也全部替换为对应id值,是不是R类就可以删除了呢?答案是肯定的,在已经完成第一种情况的优化后,这个处理的收益比较有限,因此并没有实际投入研发和使用。但是我们确实可以这么做!3资源百晓生 - resources.arsc
Resources.getIdentifier
这种间接(动态)方式获取资源id值时,也是以资源类型+名称,在resources.arsc中进行反向查找,找到后,再继续通过id值获取资源值。这个查找过程,相当于给定一个值,获取其在一个hashmap中的key。那么有没有什么方式,可以更高效实现这种运行时灵活的引用资源呢?一个比较自然的想法,是通过java反射获取R.<type>.<name>
值,那么问题来了,相对于Resources.getIdentifier
方式获取,哪种性能更好一些?答案可能并不是简单的二选一,耗时可能与资源数量,以及是否第一次查询同一种类型资源,都有关系,答案就留给读者来思考和验证吧。com.youku.android:aln:1.9.49
|-- string/m_mode
|-- layout/pager_last
|-- dimen/h_n_bar_pop_star
|-- asset/config/custom_config.json
com.youku.android:YHP:1.23.511.1
|-- layout/channel_list_footer
|-- layout/f_cover_s_feed_item
|-- drawable-night-xhdpi-v8/ic_ho
[conflict] drawable/al_down_arrow
|-- xhdpi-v4
| |-- md5:cc2ef446bf586b03fd08332a5a75b304 (com.ali.user.sdk:au:4.10.6.18)
| |-- md5:5f9c59ec3fba027c5783120effa12789 (com.ta.android:lo4android:4.10.6.18)
[conflict] string/str_retry
|-- en
| |-- not calculated (com.ali.android.phone:bee-build:10.2.3.358)
|-- default
| |-- 重试 (com.ali.android.phone:photo-build:10.2.3.57)
| |-- 点击重试 (com.ali.android.phone:bee-build:10.2.3.358)
java代码。通过R.resourceType.resourceName方式引用,例如R.string.app_name;或者通过资源id方式,直接引用,例如0x7fxxxxxx; 清单文件AndroidManifest.xml; 其它资源。
project:app:1.0
|-- array/planets_array
|-- color/white
|-- drawable/fake_drawable
|-- layout/layout_miss_view
|-- raw/app_resource_raw_chinese_text
|-- string/string_resource_chinese_name
|-- xml/app_resource_xml_chinese_text
project:library-aar-1:1.0
|-- layout/layout_contain_merge
|-- string/library_aar_1_name
Resource Reference Graph:
array:planets_array:2130771968 is reachable: false. The references =>
attr:attr_enum:2130837504 is reachable: true. The references =>
referenced by code : [com/example/libraryaar1/CustomImageView (project:library-aar-1:1.0)]
referenced by resource : [layout:layout_use_declare_styleable1:2131099652]
attr:attr_integer:2130837505 is reachable: true. The references =>
referenced by resource : [style:CustomTextStyle:2131361792]
无用资源治理情况
* [ignored] layout-xxxhdpi/layout_include_layout (project:library-aar-1:1.0)
|-- com.example.libraryaar1.NonExistCustomView
* layout/layout_miss_view (project:app:1.0, project:library-aar-1:1.0)
|-- com.example.myapplication.NonExistView2
|-- com.example.myapplication.NonExistView
# layout中引用不存在的class,在apk编译过程中,并不会引发构建失败,但依然会生成相对应的keep规则。
# 这个layout一旦在运行时被“加载“,那么会引发Java类找不到的异常。
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent">
<com.example.myapplication.NonExistView
android:id="@+id/main_textview"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!"/>
</LinearLayout>
# 生成的keep规则为:-keep class com.example.myapplication.NonExistView { <init> ( ... ) ; }
易冗余。多处资源使用同一文本时,会导致存在多份此文本; 不灵活。当线上版本出现问题时(例如各类运营活动),难以动态修改; 低安全。一些敏感信息,如果以明文硬编码文本形式存在,非常容易被获取,并用于不正当用途。
project:app:1.0
|-- array/planets_array
| |-- [text] string-array包含的中文item
|-- raw/app_resource_raw_chinese_text
| |-- [text] <files-path name="我是raw类型xml资源文件中,包含的中文文本" path="game-bundles/" />
|-- string/string_resource_chinese_name
| |-- [text] 我是中文string资源
|-- xml/app_resource_xml_chinese_text
| |-- [text] <files-path name="我是xml资源中的中文文本" path="game-bundles/" />
|-- layout/activity_main
| |-- [text] android:text="你好,世界!" />
project:library-aar-1:1.0
|-- asset/library_aar_1_asset_chinese_text
| |-- [text] 我是包含中文文本的asset资源文件.
参考文档
【google】应用资源:https://developer.android.com/guide/topics/resources/providing-resources 【google】AAPT2:https://developer.android.com/studio/command-line/aapt2