【问题标题】:Android Build Tools keep ProGuard from removing unused appcompat library classesAndroid Build Tools 阻止 ProGuard 删除未使用的 appcompat 库类
【发布时间】:2015-02-26 07:17:19
【问题描述】:

这是我build.gradle的一部分:

android {
    buildTypes {
        release {
            minifyEnabled true
            shrinkResources true
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile 'com.android.support:appcompat-v7:21.0.3'
}

proguard-rules.pro 中没有任何内容,但我注意到生成了一个名为 aapt_rules.txt 的文件,其中包含以下内容:

# view res/layout/abc_action_menu_item_layout.xml #generated:17
-keep class android.support.v7.internal.view.menu.ActionMenuItemView { <init>(...); }

# view res/layout/abc_expanded_menu_layout.xml #generated:17
-keep class android.support.v7.internal.view.menu.ExpandedMenuView { <init>(...); }

# view res/layout/abc_list_menu_item_layout.xml #generated:17
# view res/layout/abc_popup_menu_item_layout.xml #generated:17
-keep class android.support.v7.internal.view.menu.ListMenuItemView { <init>(...); }

# view res/layout/abc_screen_toolbar.xml #generated:27
-keep class android.support.v7.internal.widget.ActionBarContainer { <init>(...); }

# view res/layout/abc_action_mode_bar.xml #generated:19
# view res/layout/abc_screen_toolbar.xml #generated:43
-keep class android.support.v7.internal.widget.ActionBarContextView { <init>(...); }

# view res/layout/abc_screen_toolbar.xml #generated:17
-keep class android.support.v7.internal.widget.ActionBarOverlayLayout { <init>(...); }

# view res/layout/abc_screen_content_include.xml #generated:19
-keep class android.support.v7.internal.widget.ContentFrameLayout { <init>(...); }

# view res/layout/abc_screen_simple_overlay_action_mode.xml #generated:23
-keep class android.support.v7.internal.widget.FitWindowsFrameLayout { <init>(...); }

# view res/layout/abc_screen_simple.xml #generated:17
-keep class android.support.v7.internal.widget.FitWindowsLinearLayout { <init>(...); }

# view v11/res/layout-v11/abc_screen_content_include.xml #generated:19
-keep class android.support.v7.internal.widget.NativeActionModeAwareLayout { <init>(...); }

# view res/layout/abc_action_mode_close_item_material.xml #generated:17
# view res/layout/abc_search_dropdown_item_icons_2line.xml #generated:27
# view res/layout/abc_search_dropdown_item_icons_2line.xml #generated:37
# view res/layout/abc_search_dropdown_item_icons_2line.xml #generated:48
# view res/layout/abc_search_view.xml #generated:116
# view res/layout/abc_search_view.xml #generated:128
# view res/layout/abc_search_view.xml #generated:38
# view res/layout/abc_search_view.xml #generated:60
# view res/layout/abc_search_view.xml #generated:97
-keep class android.support.v7.internal.widget.TintImageView { <init>(...); }

# view res/layout/abc_screen_simple.xml #generated:25
# view res/layout/abc_screen_simple_overlay_action_mode.xml #generated:32
-keep class android.support.v7.internal.widget.ViewStubCompat { <init>(...); }

# view res/layout/abc_action_menu_layout.xml #generated:17
-keep class android.support.v7.widget.ActionMenuView { <init>(...); }

# view res/layout/abc_activity_chooser_view.xml #generated:19
-keep class android.support.v7.widget.LinearLayoutCompat { <init>(...); }

# view res/layout/abc_search_view.xml #generated:78
-keep class android.support.v7.widget.SearchView$SearchAutoComplete { <init>(...); }

# view res/layout/abc_screen_toolbar.xml #generated:36
-keep class android.support.v7.widget.Toolbar { <init>(...); }

这显然将 XML 使用的所有内容保留在支持库中,即使这些 XML 根本没有被使用。并且再次保留的代码可以防止使用shrinkResources true 跳过所有资源。即使没有使用支持库中的任何内容,apk 的大小也增加了很多。其他库也是如此。

那么有没有办法自定义aapt_rules.txt 或做类似的事情来删除那些未使用的代码和资源? (或者我需要在某个地方为此打开一个问题吗?)

【问题讨论】:

    标签: android gradle proguard


    【解决方案1】:

    可以看到 aapt_rules.txt 中有 cmets。在每个保留的类旁边都有引用该类的相应布局文件。像这样:

    # view res/layout/abc_list_menu_item_layout.xml #generated:17
    # view res/layout/abc_popup_menu_item_layout.xml #generated:17
    -keep class android.support.v7.internal.view.menu.ListMenuItemView { <init>(...); }
    

    如果您从构建过程中删除布局文件,该行将消失并且类将不会保留。如果该类没有在某处实际使用,则该类将被缩小。

    那么我们如何从 appcompat 库中移除布局文件呢?我可以看到几个选项,它们都不是完美的,但它们有效。

    1. 您可以从 sdk\extras\android\m2repository\com\android\support\appcompat-v7\version\appcompat-v7-version.aar 中删除文件。足以用于测试,但不适合生产,因为相同的文件可能会在其他一些项目中使用。我试过了,效果很好。

    2. 将具有相同名称的假文件放入您的项目中。会发生名称冲突。构建过程将更喜欢您的假文件,因为项目文件具有更高的优先级。这样来自 appcompat 的文件将被忽略。我试过了,效果很好。

    3. 也许您可以制作一些精美的 gradle 脚本,在构建过程中删除不需要的文件。我没试过。

    (shrinkResources 选项没有帮助,因为 aapt_rules.txt 是在实际涉及 shrinkResources 之前生成的。)

    我希望有人能提出更好的方法来做到这一点

    在这样做之后,所有不需要的行都从 aapt_rules.txt 中消失了。但它从最终的 apk 大小中节省了大约 100 KB。所以对我来说没什么大不了的。但在您的情况下,结果可能会有所不同。

    【讨论】:

    • 第二个听起来像是一个解决方案。 :-)
    • 顺便问一下,您尝试过最新的 Android 构建工具吗?他们说they've fixed it
    • @Mygod 不确定它们的意思。同样的问题似乎仍然存在。相同的未使用文件仍会进入 aapt_rules.txt。刚刚尝试使用最新的构建工具 - SDK 工具 25.1.7、构建工具 24、Android Gradle 插件 2.2.0-alpha4、appcompat 库 23.3.0。
    • 也许您必须使用 Android Studio 才能启用此功能?
    • @Mygod 我用的是 Android Studio,还是一样。
    【解决方案2】:

    如果您使用sbt-android 构建,您可以像这样手动覆盖proguardConfig

    proguardConfig in Android := List("-dontobfuscate",
      "-dontoptimize",
      "-renamesourcefileattribute SourceFile",
      "-keepattributes SourceFile,LineNumberTable",
      "-verbose",
      "-flattenpackagehierarchy",
      "-dontusemixedcaseclassnames",
      "-dontpreverify",
      "-optimizations !code/simplification/arithmetic,!code/simplification/cast,!field/*,!class/merging/*,!code/allocation/variable",
      "-keepattributes *Annotation*",
      "-dontnote android.annotation.**",
      "-dontwarn android.support.**",
      "-dontnote android.support.**",
      "-dontnote scala.ScalaObject",
      "-dontnote org.xml.sax.EntityResolver",
      "-dontnote scala.concurrent.forkjoin.**",
      "-dontwarn scala.beans.ScalaBeanInfo",
      "-dontwarn scala.concurrent.**",
      "-dontnote scala.reflect.**",
      "-dontwarn scala.reflect.**",
      "-dontwarn scala.sys.process.package$",
      "-dontwarn **$$anonfun$*",
      "-dontwarn scala.collection.immutable.RedBlack$Empty",
      "-dontwarn scala.tools.**,plugintemplate.**",
    
      "-keep class scala.collection.SeqLike { public java.lang.String toString(); }",
    
      "-keep class android.support.v7.view.menu.ListMenuItemView { <init>(...); }",
      ...
    )
    

    【讨论】:

      【解决方案3】:

      下面的定义写在getDefaultProguardFile('proguard-android.txt')中,这是{android-sdks}/tools/proguard目录下的默认proguard设置之一

       # The support library contains references to newer platform versions.
       # Don't warn about those in case this app is linking against an older
       # platform version.  We know about them, and they are safe.
       -dontwarn android.support.**
      

      所以,它保留了一切。

      无用的代码可以通过另一个 proguard 文件传递​​,{android-sdks}/tools/proguard/proguard-android-optimize.txt,默认为 5 转而优化代码(-optimizationpasses 5)。

      【讨论】:

      • 但是 dontwarn 不会保留任何东西。它只是从字面上阻止 ProGuard 警告缺少方法。
      • -keep class android.support.v4.app.** {;} -keep interface android.support.v4.app.* {;} - keep class android.support.v7.* {;} -keep interface android.support.v7.* {*;}
      • @Mygod,感谢您的正确cmets,是的,以下保留了有关支持库V4和V7的所有内容
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 2012-10-07
      • 1970-01-01
      • 2015-02-24
      • 1970-01-01
      • 2019-03-27
      • 1970-01-01
      相关资源
      最近更新 更多