首页 > 解决方案 > R8 / Android Gradle Plugin 是否足够智能以保持仅在测试 apk 中需要的库类/方法?

问题描述

假设我有一个依赖于库的应用程序。这个库有两种方法:

在我调用的应用程序usedInApp()中。我还有一个仪器测试调用usedInTest(). 如果我运行仪器测试./gradlew app:connectedDebugAndroidTest,R8/AGP 是否足够聪明,可以知道是保留usedInTest()还是删除?

标签: android-gradle-pluginandroid-r8

解决方案


由于usedInTest应用程序未使用,R8 将删除它,并且在运行测试时它将失败并显示MethodNotFouldError. 因此,您需要一个保留规则以确保usedInTest在测试时仍在应用程序中。一种好方法是添加注释,例如KeepForTesting,在应用程序中注释测试需要什么。然后添加此保留规则:

-keep,allowobfuscation class * {
  @KeepForTesting *;
}

注意allowobfuscation修改器。这允许将这些测试方法重命名为更短的名称。当 Android Studio 构建测试时,它会自动添加一个-applymapping选项,其中包含构建应用程序时生成的映射文件。这样,usedInTest在运行 R8 之后,测试中的引用将被重命名为应用程序中的实际方法。

对于 R8 本身,我们通过分析 R8 测试来发现测试中使用了 R8 的哪些部分(不在公共 API 上)来实现自动化。由此我们综合了保持规则来保持这一点。然后我们可以在 R8 上运行 R8,然后在该版本的 R8 上运行所有测试——这也是我们最终发布的版本。


推荐阅读