为了尽可能减小应用的大小并提高其运行速度,您应使用 isMinifyEnabled = true
优化和缩减发布 build。
这样做可启用缩减功能(用于移除未使用的代码);混淆功能(用于缩短应用的类和成员的名称);以及优化功能(用于应用改进的代码优化策略,以进一步减小应用的大小并提升其性能)。本页介绍了 R8 如何为您的项目执行这些编译时任务,以及如何对其进行自定义。
当您使用 Android Gradle 插件 3.4.0 或更高版本构建项目时,该插件不再使用 ProGuard 执行编译时代码优化,而是与 R8 编译器协同工作,处理以下编译时任务:
- 代码缩减(即摇树优化):从应用及其库依赖项中检测并安全地移除不使用的类、字段、方法和属性(这使其成为了一个对于规避 64k 引用限制非常有用的工具)。例如,如果您仅使用某个库依赖项的少数几个 API,那么缩减功能可以识别应用不使用的库代码并仅从应用中移除这部分代码。如需了解详情,请转到介绍如何缩减代码的部分。
- 资源缩减:从封装应用中移除不使用的资源,包括应用库依赖项中不使用的资源。此功能可与代码缩减功能结合使用,这样一来,移除不使用的代码后,也可以安全地移除不再引用的所有资源。如需了解详情,请转到介绍如何缩减资源的部分。
- 优化:检查并重写代码,以提升运行时性能并进一步减小应用的 DEX 文件的大小。这可将代码的运行时性能提高最多 30%,显著缩短启动时间和帧时间。例如,如果 R8 检测到从未采用过给定 if/else 语句的
else {}
分支,则会移除else {}
分支的代码。如需了解详情,请转到介绍代码优化的部分。 - 混淆处理(或标识符缩减):缩短类和成员的名称,从而减小 DEX 文件的大小。如需了解详情,请参阅介绍如何对代码进行混淆处理的部分。
在构建应用的发布版本时,您可以将 R8 配置为执行上述编译时任务。您还可以停用某些任务或通过 ProGuard 规则文件自定义 R8 的行为。事实上,R8 支持所有现有 ProGuard 规则文件,因此您在更新 Android Gradle 插件以使用 R8 时,无需更改现有规则。
启用缩减、混淆处理和优化功能
当您使用 Android Studio 3.4 或 Android Gradle 插件 3.4.0 及更高版本时,R8 是默认编译器,用于将项目的 Java 字节码转换为在 Android 平台上运行的 DEX 格式。不过,当您使用 Android Studio 创建新项目时,缩减、混淆处理和代码优化功能默认处于停用状态。这是因为,这些编译时优化功能会增加项目的构建时间,而且如果您没有充分自定义要保留的代码,还可能会引入错误。
因此,在构建应用的最终版本(也就是在发布应用之前测试的版本)时,最好启用这些编译时任务。如需启用缩减、混淆处理和优化功能,请在项目级 build 脚本中添加以下代码。
Kotlin
android { buildTypes { getByName("release") { // Enables code shrinking, obfuscation, and optimization for only // your project's release build type. Make sure to use a build // variant with `isDebuggable=false`. isMinifyEnabled = true // Enables resource shrinking, which is performed by the // Android Gradle plugin. isShrinkResources = true proguardFiles( // Includes the default ProGuard rules files that are packaged with // the Android Gradle plugin. To learn more, go to the section about // R8 configuration files. getDefaultProguardFile("proguard-android-optimize.txt"), // Includes a local, custom Proguard rules file "proguard-rules.pro" ) } } ... }
Groovy
android { buildTypes { release { // Enables code shrinking, obfuscation, and optimization for only // your project's release build type. Make sure to use a build // variant with `debuggable false`. minifyEnabled true // Enables resource shrinking, which is performed by the // Android Gradle plugin. shrinkResources true // Includes the default ProGuard rules files that are packaged with // the Android Gradle plugin. To learn more, go to the section about // R8 configuration files. proguardFiles getDefaultProguardFile( 'proguard-android-optimize.txt'), 'proguard-rules.pro' } } ... }
R8 配置文件
R8 使用 ProGuard 规则文件来修改其默认行为并更好地了解应用的结构,比如充当应用代码入口点的类。虽然您可以修改其中一些规则文件,但某些规则可能由编译时工具(如 AAPT2)自动生成,或从应用的库依赖项继承而来。下表介绍了 R8 使用的 ProGuard 规则文件的来源。
来源 | 位置 | 说明 |
Android Studio | <module-dir>/proguard-rules.pro
|
当您使用 Android Studio 创建新模块时,Android Studio 会在该模块的根目录中创建 proguard-rules.pro 文件。
默认情况下,此文件不会应用任何规则。因此,请在此处添加您自己的 ProGuard 规则,比如自定义保留规则。 |
Android Gradle 插件 | 由 Android Gradle 插件在编译时生成。 | Android Gradle 插件会生成 proguard-android-optimize.txt (其中包含了对大多数 Android 项目都有用的规则),并启用 @Keep* 注解。
默认情况下,使用 Android Studio 创建新模块时,模块级 build 脚本会将此规则文件添加到您的发布 build 中。
注意:虽然 Android Gradle 插件包含额外的预定义 ProGuard 规则文件,但建议您使用 |
库依赖项 |
在 AAR 库中:
在 JAR 库中: 除了这些位置之外,Android Gradle 插件 3.6 或更高版本还支持有针对性缩减规则。 |
如果某个 AAR 或 JAR 库是使用它自己的规则文件发布的,并且您将该库作为编译时依赖项纳入到项目中,那么 R8 在编译项目时会自动应用这些规则。 除了传统的 ProGuard 规则之外,Android Gradle 插件 3.6 或更高版本还支持目标缩减规则。这些规则针对特定缩减器(R8 或 ProGuard)以及特定缩减器版本。 如果库需要某些规则才能正常运行,则使用库随附的规则文件非常有用。也就是说,库开发者已经为您执行了问题排查步骤。 不过,您应该知道,因为这些规则是累加的,所以库依赖项包含的某些规则无法移除,并且可能会影响应用其他部分的编译。例如,如果某个库包含停用代码优化的规则,该规则将针对整个项目停用优化。 |
Android 资源打包工具 2 (AAPT2) | 使用 minifyEnabled true 构建项目后:<module-dir>/build/intermediates/aapt_proguard_file/.../aapt_rules.txt
|
AAPT2 会根据对应用清单中的类、布局及其他应用资源的引用,生成保留规则。例如,AAPT2 会为您在应用清单中注册为入口点的每个 activity 添加一个保留规则。 |
自定义配置文件 | 默认情况下,当您使用 Android Studio 创建新模块时,IDE 会创建 <module-dir>/proguard-rules.pro ,以便您添加自己的规则。
|
您可以添加其他配置,R8 会在编译时应用这些配置。 |
如果您将 minifyEnabled
属性设为 true
,R8 会将来自上述所有可用来源的规则组合在一起。在您排查 R8 问题时需要谨记这一点,因为其他编译时依赖项(如库依赖项)可能会引入您不了解的 R8 行为变化。
如需输出 R8 在构建项目时应用的所有规则的完整报告,请将以下代码添加到模块的 proguard-rules.pro
文件中:
// You can specify any path and filename.
-printconfiguration ~/tmp/full-r8-config.txt
有针对性缩减的规则
Android Gradle 插件 3.6 或更高版本支持定位到特定缩减工具(R8 或 ProGuard)以及特定缩减工具版本的库规则。这样,库开发者就可以根据自己的需求调整规则,使其在使用新缩减工具版本的项目中发挥最佳效果,同时允许在使用旧版缩减工具版本的项目中继续使用现有规则。
如需指定目标缩减规则,库开发者需要在 AAR 或 JAR 库内的特定位置添加这些规则,如下所述。
In an AAR library:
proguard.txt (legacy location)
classes.jar
└── META-INF
└── com.android.tools (targeted shrink rules location)
├── r8-from-<X>-upto-<Y>/<R8-rules-file>
└── proguard-from-<X>-upto-<Y>/<ProGuard-rules-file>
In a JAR library:
META-INF
├── proguard/<ProGuard-rules-file> (legacy location)
└── com.android.tools (targeted shrink rules location)
├── r8-from-<X>-upto-<Y>/<R8-rules-file>
└── proguard-from-<X>-upto-<Y>/<ProGuard-rules-file>
这意味着,目标缩减规则存储在 JAR 的 META-INF/com.android.tools
目录中或 AAR 的 classes.jar
内的 META-INF/com.android.tools
目录中。
在该目录下,可以有多个名称为 r8-from-<X>-upto-<Y>
或 proguard-from-<X>-upto-<Y>
的目录,以指明目录中的规则是针对哪个版本的缩减器编写的。请注意,-from-<X>
和 -upto-<Y>
部分是可选的,<Y>
版本是排他的,并且版本范围必须连续。
例如,r8-upto-8.0.0
、r8-from-8.0.0-upto-8.2.0
和 r8-from-8.2.0
构成了一组有效的目标缩减规则。R8 将使用 r8-from-8.0.0-upto-8.2.0
目录下的规则,从 8.0.0 版到 8.2.0 版(不包括 8.2.0 版)。
有了这些信息,Android Gradle 插件 3.6 或更高版本将从匹配的 R8 目录中选择规则。如果库未指定目标缩减规则,Android Gradle 插件将从旧版位置(对于 AAR 为 proguard.txt
,对于 JAR 为 META-INF/proguard/<ProGuard-rules-file>
)中选择规则。
库开发者可以选择在其库中添加目标缩减规则或旧版 ProGuard 规则,如果他们希望与版本低于 3.6 的 Android Gradle 插件或其他工具保持兼容性,则可以同时添加这两种规则。
添加其他配置
当您使用 Android Studio 创建新项目或模块时,该 IDE 会创建一个 <module-dir>/proguard-rules.pro
文件,以便您添加自己的规则。此外,您还可以通过将相应文件添加到模块的 build 脚本的 proguardFiles
属性中,从其他文件添加额外的规则。
例如,您可以通过在相应的 productFlavor
代码块中再添加一个 proguardFiles
属性来添加每个 build 变体专用的规则。以下 Gradle 文件会将 flavor2-rules.pro
添加到 flavor2
产品变种中。现在,flavor2
使用全部三个 ProGuard 规则,因为还应用了来自 release
代码块的规则。
此外,您还可以添加 testProguardFiles
属性,用于指定仅包含在测试 APK 中的 ProGuard 文件列表:
Kotlin
android { ... buildTypes { getByName("release") { isMinifyEnabled = true proguardFiles( getDefaultProguardFile("proguard-android-optimize.txt"), // List additional ProGuard rules for the given build type here. By default, // Android Studio creates and includes an empty rules file for you (located // at the root directory of each module). "proguard-rules.pro" ) testProguardFiles( // The proguard files listed here are included in the // test APK only. "test-proguard-rules.pro" ) } } flavorDimensions.add("version") productFlavors { create("flavor1") { ... } create("flavor2") { proguardFile("flavor2-rules.pro") } } }
Groovy
android { ... buildTypes { release { minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), // List additional ProGuard rules for the given build type here. By default, // Android Studio creates and includes an empty rules file for you (located // at the root directory of each module). 'proguard-rules.pro' testProguardFiles // The proguard files listed here are included in the // test APK only. 'test-proguard-rules.pro' } } flavorDimensions "version" productFlavors { flavor1 { ... } flavor2 { proguardFile 'flavor2-rules.pro' } } }
缩减代码
如果将 minifyEnabled
属性设为 true
,系统会默认启用 R8 代码缩减功能。
代码缩减(也称为“摇树优化”)是指移除 R8 确定在运行时不需要的代码的过程。此过程可以大大减小应用的大小,例如,当您的应用包含许多库依赖项,但只使用它们的一小部分功能时。
为了缩减应用的代码,R8 首先会根据组合的配置文件集确定应用代码的所有入口点。这些入口点包括 Android 平台可用来打开应用的 activity 或服务的所有类。从每个入口点开始,R8 会检查应用的代码来构建一张图表,列出应用在运行时可能会访问的所有方法、成员变量和其他类。系统会将与该图表没有关联的代码视为执行不到的代码,并可能会从应用中移除该代码。
图 1 显示了一个具有运行时库依赖项的应用。R8 通过检查应用的代码,确定可以从 MainActivity.class
入口点执行到的 foo()
、faz()
和 bar()
方法。不过,您的应用从未在运行时使用过 OkayApi.class
类或其 baz()
方法,因此 R8 会在缩减应用时移除该代码。
R8 通过项目的 R8 配置文件中的 -keep
规则确定入口点。也就是说,保留规则指定 R8 在缩减应用时不应舍弃的类,R8 将这些类视为应用的可能入口点。Android Gradle 插件和 AAPT2 会自动为您生成大多数应用项目(如应用的 activity、视图和服务)所需的保留规则。不过,如果您需要使用其他保留规则来自定义此默认行为,请参阅介绍如何自定义要保留的代码的部分。
如果您只想减小应用资源的大小,请跳到介绍如何缩减资源的部分。
请注意,如果库项目缩减,依赖于该库的应用会包含缩减的库类。如果库 APK 中缺少类,您可能需要调整库保留规则。如果您以 AAR 格式构建和发布库,则您的库所依赖的本地 JAR 文件在 AAR 文件中不会缩小。
自定义要保留的代码
在大多数情况下,如要让 R8 仅移除不使用的代码,使用默认的 ProGuard 规则文件 (proguard-android-optimize.txt
) 就已足够。不过,在某些情况下,R8 很难做出正确判断,因而可能会移除应用实际上需要的代码。下面列举了几个示例,说明它在什么情况下可能会错误地移除代码:
- 当应用通过 Java 原生接口 (JNI) 调用方法时
- 当您的应用在运行时查询代码时(如使用反射)
通过测试应用应该可以发现因错误移除代码而导致的错误,但您也可以通过生成已移除代码的报告检查移除了哪些代码。
如需修复错误并强制 R8 保留某些代码,请在 ProGuard 规则文件中添加 -keep
代码行。例如:
-keep public class MyClass
或者,您也可以为要保留的代码添加 @Keep
注解。在类上添加 @Keep
可按原样保留整个类。在方法或字段上添加该注释,将使该方法/字段(及其名称)以及类名称保持不变。请注意,只有在使用 AndroidX 注解库且您添加 Android Gradle 插件随附的 ProGuard 规则文件时,此注解才可用。有关详情,请参阅介绍如何启用缩减功能的部分。
在使用 -keep
选项时,有许多注意事项;如需详细了解如何自定义规则文件,请参阅 ProGuard 手册。问题排查部分简要介绍了移除代码后您可能会遇到的其他常见问题。
移除原生库
默认情况下,原生代码库已从应用的发布 build 中移除。此移除操作包括移除应用所使用的所有原生库中包含的符号表及调试信息。移除原生代码库会显著缩减大小;但是,由于缺少信息(例如类和函数名称),无法诊断 Google Play 管理中心内的崩溃问题。
原生代码崩溃支持
Google Play 管理中心会在 Android Vitals 下报告原生代码崩溃问题。只需几个步骤,即可为应用生成并上传原生代码调试符号文件。此文件可在 Android Vitals 中启用经过符号化解析的原生代码崩溃堆栈轨迹(包括类和函数名称),来帮助您在生产环境中调试应用。这些步骤因项目中使用的 Android Gradle 插件版本和项目的构建输出而有所不同。
Android Gradle 插件版本 4.1 或更高版本
如果您的项目构建的是 Android App Bundle,您可以在其中自动添加原生代码调试符号文件。如需在发布 build 中添加该文件,请将以下代码添加到应用的 build.gradle.kts
文件中:
android.buildTypes.release.ndk.debugSymbolLevel = { SYMBOL_TABLE | FULL }
从以下调试符号级别中进行选择:
- 使用
SYMBOL_TABLE
在 Play 管理中心的符号化解析后的堆栈轨迹中获取函数名称。此级别支持 Tombstone。 - 使用
FULL
在 Play 管理中心的符号化解析后的堆栈轨迹中获取函数名称、文件和行号。
如果您的项目构建的是 APK,请使用之前显示的 build.gradle.kts
构建设置,以单独生成原生代码调试符号文件。手动将原生代码调试符号文件上传到 Google Play 管理中心。在构建流程中,Android Gradle 插件会在以下项目位置输出此文件:
app/build/outputs/native-debug-symbols/variant-name/native-debug-symbols.zip
Android Gradle 插件版本 4.0 或更低版本(及其他构建系统)
在构建流程中,Android Gradle 插件会在项目目录中保留未移除的库的副本。该目录结构类似于以下结构:
app/build/intermediates/cmake/universal/release/obj/
├── armeabi-v7a/
│ ├── libgameengine.so
│ ├── libothercode.so
│ └── libvideocodec.so
├── arm64-v8a/
│ ├── libgameengine.so
│ ├── libothercode.so
│ └── libvideocodec.so
├── x86/
│ ├── libgameengine.so
│ ├── libothercode.so
│ └── libvideocodec.so
└── x86_64/
├── libgameengine.so
├── libothercode.so
└── libvideocodec.so
压缩以下目录的内容:
cd app/build/intermediates/cmake/universal/release/obj
zip -r symbols.zip .
手动将
symbols.zip
文件上传到 Google Play 管理中心。
缩减资源
资源缩减只有在与代码缩减配合使用时才能发挥作用。在代码缩减器移除所有不使用的代码后,资源缩减器便可确定应用仍要使用的资源,当您添加包含资源的代码库时尤其如此。您必须移除不使用的库代码,使库资源变为未引用资源,因而可由资源缩减器移除。
如需启用资源缩减功能,请将 build 脚本中的 shrinkResources
属性(若为代码缩减,则还包括 minifyEnabled
)设为 true
。例如:
Kotlin
android { ... buildTypes { getByName("release") { isShrinkResources = true isMinifyEnabled = true proguardFiles( getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro" ) } } }
Groovy
android { ... buildTypes { release { shrinkResources true minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } }
如果您尚未使用用于缩减代码的 minifyEnabled
构建应用,请先尝试使用它,然后再启用 shrinkResources
,因为您可能需要先修改 proguard-rules.pro
文件以保留动态创建或调用的类或方法,然后再开始移除资源。
自定义要保留的资源
如果您有想要保留或舍弃的特定资源,请在项目中创建一个包含 <resources>
标记的 XML 文件,并在 tools:keep
属性中指定每个要保留的资源,在 tools:discard
属性中指定每个要舍弃的资源。这两个属性都接受以逗号分隔的资源名称列表。您可以将星号字符用作通配符。
例如:
<?xml version="1.0" encoding="utf-8"?> <resources xmlns:tools="http://schemas.android.com/tools" tools:keep="@layout/l_used*_c,@layout/l_used_a,@layout/l_used_b*" tools:discard="@layout/unused2" />
将该文件保存在项目资源中,例如,保存在 res/raw/my.package.keep.xml
中。构建系统不会将此文件打包到应用中。
注意:请务必为 keep
文件使用唯一名称。否则,当不同的库关联在一起时,其 keep 规则会发生冲突,从而导致被忽略的规则或不需要保留的资源可能出现问题。
指定要舍弃的资源可能看似没有必要,因为您本可将其删除,但在使用 build 变体时,这样做可能很有用。例如,如果您知道给定资源似乎正用于代码(因此不会被缩减器移除),但知道它实际上不会用于给定的 build 变体,则可以将所有资源放入通用项目目录中,然后为每个 build 变体分别创建一个不同的 my.package.build.variant.keep.xml
文件。构建工具也可能会错误地将某个资源标识为必需资源,这是因为编译器会内嵌添加资源 ID,而资源分析器可能不知道真正引用的资源与代码中恰好具有相同值的整数值之间的区别。
启用严格引用检查
通常,资源缩减器可以准确地判断是否使用了某个资源。不过,如果您的代码会调用
Resources.getIdentifier()
(或者您的任何库会执行此调用,例如 AppCompat 库便会执行此调用),这意味着您的代码将根据动态生成的字符串查询资源名称。当您启用严格引用检查时,资源缩减器在默认情况下会采取保护行为,将所有具有匹配名称格式的资源标记为可能已使用,无法移除。
例如,以下代码会将所有带 img_
前缀的资源标记为已使用。
Kotlin
val name = String.format("img_%1d", angle + 1) val res = resources.getIdentifier(name, "drawable", packageName)
Java
String name = String.format("img_%1d", angle + 1); res = getResources().getIdentifier(name, "drawable", getPackageName());
资源缩减器还会查看代码中的所有字符串常量以及各种 res/raw/
资源,以查找格式类似于 file:///android_res/drawable//ic_plus_anim_016.png
的资源网址。如果它找到与此类似的字符串,或找到其他看似可用来构建与此类似的网址的字符串,则不会将它们移除。
这些是默认情况下启用的安全缩减模式的示例。不过,您可以停用这种“防患于未然”的处理方式,指定资源缩减器只保留确定要使用的资源。为此,您可以将 keep.xml
文件中的 shrinkMode
设为 strict
,如下所示:
<?xml version="1.0" encoding="utf-8"?> <resources xmlns:tools="http://schemas.android.com/tools" tools:shrinkMode="strict" />
如果您确实启用了严格缩减模式,并且您的代码也通过动态生成的字符串引用资源(如上所示),那么您必须使用 tools:keep
属性手动保留这些资源。
移除未使用的备用资源
Gradle 资源缩减器只会移除未由应用代码引用的资源,这意味着,它不会移除用于不同设备配置的备用资源。如有必要,您可以使用 Android Gradle 插件的 resConfigs
属性移除应用不需要的备用资源文件。
例如,如果您使用的是包含语言资源的库(如 AppCompat 或 Google Play 服务),那么您的应用中将包含这些库中消息的所有已翻译语言的字符串,而无论应用的其余部分是否已翻译为相同的语言。如果您只想保留应用正式支持的语言,可以使用 resConfig
属性指定这些语言。系统会移除未指定语言的所有资源。
以下代码段展示了如何设置只保留英语和法语的语言资源:
Kotlin
android { defaultConfig { ... resourceConfigurations.addAll(listOf("en", "fr")) } }
Groovy
android { defaultConfig { ... resConfigs "en", "fr" } }
如果使用 Android App Bundle 格式发布应用,那么在默认情况下,安装该应用时只会下载用户设备上配置的语言的应用版本。同样,下载内容中仅包含与设备的屏幕密度相匹配的资源以及与设备的 ABI 相匹配的原生库。如需了解详情,请参阅 Android App Bundle 配置。
对于使用 APK 发布的旧版应用(创建于 2021 年 8 月之前),您可以通过构建多个 APK 并使每个 APK 对应不同的设备配置,自定义要包含在 APK 中的屏幕密度或 ABI 资源。
合并重复资源
默认情况下,Gradle 还会合并同名的资源,如可能位于不同资源文件夹中的同名可绘制对象。这一行为不受 shrinkResources
属性控制,也无法停用,因为当多个资源与代码查询的名称匹配时,有必要利用这一行为避免错误。
只有在两个或更多个文件具有完全相同的资源名称、类型和限定符时,才会进行资源合并。Gradle 会在重复项中选择它认为最合适的文件(根据下述优先顺序),并且只将这一个资源传递给 AAPT,以便在最终工件中分发。
Gradle 会在以下位置查找重复资源:
- 与主源代码集关联的主资源,一般位于
src/main/res/
中。 - 变体叠加,来自 build 类型和 build 变种。
- 库项目依赖项。
Gradle 会按以下级联优先顺序合并重复资源:
依赖项 → 主资源 → build 变种 → build 类型
例如,如果某个重复资源同时出现在主资源和 build 变种中,Gradle 会选择 build 变种中的资源。
如果完全相同的资源出现在同一源代码集中,Gradle 无法合并它们,并且会发出资源合并错误。如果您在 build.gradle.kts
文件的 sourceSet
属性中定义了多个源代码集,就可能会发生这种情况。例如,如果 src/main/res/
和 src/main/res2/
包含完全相同的资源,就可能会发生这种情况。
对代码进行混淆处理
混淆处理的目的是通过缩短应用的类、方法和字段的名称来缩减应用的大小。下面是使用 R8 进行混淆处理的一个示例:
androidx.appcompat.app.ActionBarDrawerToggle$DelegateProvider -> a.a.a.b:
androidx.appcompat.app.AlertController -> androidx.appcompat.app.AlertController:
android.content.Context mContext -> a
int mListItemLayout -> O
int mViewSpacingRight -> l
android.widget.Button mButtonNeutral -> w
int mMultiChoiceItemLayout -> M
boolean mShowTitle -> P
int mViewSpacingLeft -> j
int mButtonPanelSideLayout -> K
虽然混淆处理不会从应用中移除代码,但如果应用的 DEX 文件将许多类、方法和字段编入索引,那么混淆处理将可以显著缩减应用的大小。不过,由于混淆处理会对代码的不同部分进行重命名,因此在执行某些任务(如检查堆栈轨迹)时需要使用额外的工具。如需了解混淆处理后的堆栈轨迹,请参阅介绍如何解码经过混淆处理的堆栈轨迹的部分。
此外,如果您的代码依赖于应用的方法和类的可预测命名(例如,使用反射时),您应该将相应签名视为入口点并为其指定保留规则,如介绍如何自定义要保留的代码的部分中所述。这些保留规则会告知 R8 不仅要在应用的最终 DEX 中保留该代码,而且还要保留其原始命名。
解码经过混淆处理的堆栈轨迹
R8 对代码进行混淆处理后,理解堆栈轨迹的难度将会极大增加,因为类和方法的名称可能有变化。如需获取原始堆栈轨迹,您应对堆栈轨迹进行轨迹还原。
代码优化
为了进一步优化应用,R8 会在更深的层次上检查代码,以移除更多不使用的代码,或者在可能的情况下重写代码,以使其更简洁。下面是此类优化的几个示例:
- 如果您的代码从未采用过给定 if/else 语句的
else {}
分支,R8 可能会移除else {}
分支的代码。 - 如果您的代码仅在少数几个位置调用某个方法,R8 可能会移除该方法并将其内嵌到少数几个调用点。
- 如果 R8 确定某个类只有一个唯一子类且该类本身未实例化(例如,一个仅由一个具体实现类使用的抽象基类),它就可以将这两个类组合在一起并从应用中移除一个类。
- 如需了解详情,请阅读 Jake Wharton 撰写的关于 R8 优化的博文。
R8 不允许您停用或启用离散优化,也不允许您修改优化的行为。事实上,R8 会忽略尝试修改默认优化的任何 ProGuard 规则,例如 -optimizations
和 -optimizationpasses
。此限制很重要,因为随着 R8 的不断改进,维护标准的优化行为有助于 Android Studio 团队轻松排查并解决您可能遇到的任何问题。
请注意,启用优化将更改应用的堆栈轨迹。例如,进行内嵌会移除堆栈帧。如需了解如何获取原始堆栈轨迹,请参阅轨迹还原部分。
对运行时性能的影响
如果同时启用缩减、混淆和优化,R8 最多可将代码的运行时性能(包括界面线程上的启动时间和帧时间)提升 30%。停用其中任何一项都会大大限制 R8 使用的优化集。
如果启用了 R8,您还应创建启动配置文件,以进一步提升启动性能。
启用增强型优化
R8 包含一组额外的优化功能(称为“完整模式”),这使得它的行为与 ProGuard 不同。从 Android Gradle 插件版本 8.0.0 开始,这些优化默认处于启用状态。
您可以通过在项目的 gradle.properties
文件中添加以下代码来停用这些额外的优化功能:
android.enableR8.fullMode=false
由于这些额外的优化功能会使 R8 的行为与 ProGuard 不同,因此,如果您使用的是专为 ProGuard 设计的规则,这些优化功能可能会要求您添加额外的 ProGuard 规则,以避免运行时出现问题。例如,假设您的代码通过 Java Reflection API 引用一个类。不使用“完整模式”时,R8 会假设您打算在运行时检查和操纵该类的对象(即使您的代码实际上并不这样做),因此它会自动保留该类及其静态初始化程序。
不过,在使用“完整模式”时,R8 不会做出这一假设,并且如果 R8 断言您的代码永远不会在运行时使用该类,则会从应用的最终 DEX 中移除该类。也就是说,如果您想保留类及其静态初始化程序,则需要在规则文件中添加保留规则才能实现这一点。
如果您在使用 R8 的“完整模式”时遇到任何问题,请参阅 R8 常见问题解答页面,了解可能的解决方案。如果您无法解决问题,请报告错误。
对堆栈轨迹进行轨迹还原
经过 R8 处理的代码会发生各种更改,这可能会使堆栈轨迹更难以理解,因为堆栈轨迹与源代码不完全一致。如果未保留调试信息,就可能会出现行号更改的情况。这可能是由内嵌和轮廓等优化造成的。影响最大的因素是混淆处理;进行混淆处理时,就连类和方法的名称都会更改。
为了还原原始堆栈轨迹,R8 提供了 retrace 命令行工具,该工具与命令行工具软件包捆绑在一起。
如需支持对应用的堆栈轨迹进行轨迹还原,您应通过向模块的 proguard-rules.pro
文件添加以下规则来确保 build 保留足够的信息以进行轨迹还原:
-keepattributes LineNumberTable,SourceFile
-renamesourcefileattribute SourceFile
LineNumberTable
属性会在方法中保留位置信息,以便以堆栈轨迹的形式输出这些位置。SourceFile
属性可确保所有可能的运行时都实际输出位置信息。-renamesourcefileattribute
指令用于将堆栈轨迹中的源文件名称设置为仅包含 SourceFile
。在轨迹还原过程中不需要实际的原始源文件名称,因为映射文件中包含原始源文件。
R8 每次运行时都会创建一个 mapping.txt
文件,其中包含将堆栈轨迹重新映射为原始堆栈轨迹所需的信息。Android Studio 会将该文件保存在 <module-name>/build/outputs/mapping/<build-type>/
目录中。
在 Google Play 上发布应用时,您可以上传每个应用版本对应的 mapping.txt
文件。使用 Android App Bundle 格式发布应用时,系统会自动将此文件包含在 app bundle 内容中。然后,Google Play 会根据用户报告的问题对传入的堆栈轨迹进行轨迹还原,以便您可以在 Play 管理中心查看这些堆栈轨迹。如需了解详情,请参阅介绍如何对崩溃堆栈轨迹进行去混淆处理的帮助中心文章。
排查 R8 问题
本部分介绍在使用 R8 启用代码缩减、混淆和优化功能时排查问题的一些策略。如果您在下文中找不到相关问题的解决方案,您还应参阅 R8 常见问题解答页面和 ProGuard 的问题排查指南。
生成移除的(或保留的)代码的报告
为了便于排查特定的 R8 问题,建议您查看 R8 从您的应用中移除的所有代码的报告。请针对要为其生成此报告的每个模块,将 -printusage <output-dir>/usage.txt
添加到您的自定义规则文件中。当您在启用 R8 的情况下构建应用时,R8 会按照您指定的路径和文件名输出报告。移除的代码的报告与以下输出类似:
androidx.drawerlayout.R$attr
androidx.vectordrawable.R
androidx.appcompat.app.AppCompatDelegateImpl
public void setSupportActionBar(androidx.appcompat.widget.Toolbar)
public boolean hasWindowFeature(int)
public void setHandleNativeActionModesEnabled(boolean)
android.view.ViewGroup getSubDecor()
public void setLocalNightMode(int)
final androidx.appcompat.app.AppCompatDelegateImpl$AutoNightModeManager getAutoNightModeManager()
public final androidx.appcompat.app.ActionBarDrawerToggle$Delegate getDrawerToggleDelegate()
private static final boolean DEBUG
private static final java.lang.String KEY_LOCAL_NIGHT_MODE
static final java.lang.String EXCEPTION_HANDLER_MESSAGE_SUFFIX
...
如果您要查看 R8 根据项目的保留规则确定的入口点的报告,请将 -printseeds <output-dir>/seeds.txt
添加到您的自定义规则文件中。当您在启用 R8 的情况下构建应用时,R8 会按照您指定的路径和文件名输出报告。保留的入口点的报告与以下输出类似:
com.example.myapplication.MainActivity
androidx.appcompat.R$layout: int abc_action_menu_item_layout
androidx.appcompat.R$attr: int activityChooserViewStyle
androidx.appcompat.R$styleable: int MenuItem_android_id
androidx.appcompat.R$styleable: int[] CoordinatorLayout_Layout
androidx.lifecycle.FullLifecycleObserverAdapter
...
排查资源缩减问题
当您缩减资源时,Build 图标 窗口会显示从应用中移除的资源的摘要。(您需要先点击窗口左侧的 Toggle view 图标 以显示 Gradle 的详细文本输出。)例如:
:android:shrinkDebugResources
Removed unused resources: Resource data reduced from 2570KB to 1711KB: Removed 33%
:android:validateDebugSigning
Gradle 还会在 <module-name>/build/outputs/mapping/release/
(ProGuard 输出文件所在的文件夹)中创建一个名为 resources.txt
的诊断文件。此文件包含一些详细信息,比如,哪些资源引用了其他资源,哪些资源在使用,哪些资源被移除。
例如,如需了解您的应用为何仍包含 @drawable/ic_plus_anim_016
,请打开 resources.txt
文件并搜索该文件名。您可能会发现,有其他资源引用了它,如下所示:
16:25:48.005 [QUIET] [system.out] @drawable/add_schedule_fab_icon_anim : reachable=true
16:25:48.009 [QUIET] [system.out] @drawable/ic_plus_anim_016
您现在需要知道为什么可执行到 @drawable/add_schedule_fab_icon_anim
,如果您向上搜索,就会发现“The root reachable resources are:”下列有该资源。这意味着存在对 add_schedule_fab_icon_anim
的代码引用(即在可执行到的代码中找到了其 R.drawable ID)。
如果您使用的不是严格检查,当存在看似可用于为动态加载资源构建资源名称的字符串常量时,就可将资源 ID 标记为“可执行到”。在这种情况下,如果您在构建输出中搜索资源名称,可能会发现如下消息:
10:32:50.590 [QUIET] [system.out] Marking drawable:ic_plus_anim_016:2130837506
used because it format-string matches string pool constant ic_plus_anim_%1$d.
如果您看到一个这样的字符串,并且确定该字符串未用于动态加载给定资源,就可以使用 tools:discard
属性通知构建系统将其移除,具体如介绍如何自定义要保留的资源部分中所述。