Android 构建工具阻止 ProGuard 删除未使用的 appcompat 库 类
Android Build Tools keep ProGuard from removing unused appcompat library classes
这是我的一部分 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
或做类似的事情来删除那些未使用的代码和资源? (或者我需要在某个地方为此打开一个问题吗?)
以下定义写在getDefaultProguardFile('proguard-android.txt')中,它是{android-sdks}/tools/proguard[=11目录下的默认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)。
您可以看到aapt_rules.txt中有评论。在每个保留的 class 旁边,有相应的布局文件引用了这个 class。像这样:
# 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>(...); }
如果您从构建过程中删除布局文件,此行将消失并且 class 将不会保留。 class 如果没有在某处实际使用,将被缩小。
那么我们如何从 appcompat 库中删除布局文件呢?我可以看到几个选项,其中 none 是完美的,但它们有效。
您可以从 sdk\extras\android\m2repository\com\android\support\appcompat-v7\version\appcompat-v7-version.aar 中删除文件。足够用于测试,但不适合生产,因为同一个文件可能会在其他一些项目中使用。我试过了,成功了。
将同名的假文件放入您的项目中。名称冲突将发生。构建过程将更喜欢您的假文件,因为项目文件具有更高的优先级。这样来自 appcompat 的文件将被忽略。我试过了,成功了。
也许您可以编写一些花哨的 gradle 脚本来在构建过程中删除不需要的文件。我没试过。
(shrinkResources 选项没有帮助,因为 aapt_rules.txt 是在实际涉及 shrinkResources 之前生成的。)
我希望有人能提出更好的方法
这样做之后,所有不需要的行都从 aapt_rules.txt 中消失了。但它比最终的 apk 大小节省了大约 100 KB。所以对我来说没什么大不了的。但在您的情况下,结果可能会有所不同。
如果您使用 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>(...); }",
...
)
这是我的一部分 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
或做类似的事情来删除那些未使用的代码和资源? (或者我需要在某个地方为此打开一个问题吗?)
以下定义写在getDefaultProguardFile('proguard-android.txt')中,它是{android-sdks}/tools/proguard[=11目录下的默认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)。
您可以看到aapt_rules.txt中有评论。在每个保留的 class 旁边,有相应的布局文件引用了这个 class。像这样:
# 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>(...); }
如果您从构建过程中删除布局文件,此行将消失并且 class 将不会保留。 class 如果没有在某处实际使用,将被缩小。
那么我们如何从 appcompat 库中删除布局文件呢?我可以看到几个选项,其中 none 是完美的,但它们有效。
您可以从 sdk\extras\android\m2repository\com\android\support\appcompat-v7\version\appcompat-v7-version.aar 中删除文件。足够用于测试,但不适合生产,因为同一个文件可能会在其他一些项目中使用。我试过了,成功了。
将同名的假文件放入您的项目中。名称冲突将发生。构建过程将更喜欢您的假文件,因为项目文件具有更高的优先级。这样来自 appcompat 的文件将被忽略。我试过了,成功了。
也许您可以编写一些花哨的 gradle 脚本来在构建过程中删除不需要的文件。我没试过。
(shrinkResources 选项没有帮助,因为 aapt_rules.txt 是在实际涉及 shrinkResources 之前生成的。)
我希望有人能提出更好的方法
这样做之后,所有不需要的行都从 aapt_rules.txt 中消失了。但它比最终的 apk 大小节省了大约 100 KB。所以对我来说没什么大不了的。但在您的情况下,结果可能会有所不同。
如果您使用 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>(...); }",
...
)