Android 的 multidex 可以使用maindexlist来指定哪些包的class放在第一个dex中,但却要我们自己手动指定,而且不能使用正则或者其他通配符指定,非常不方便。
Android的gradle构建方式给自定义构建带来很大的便利。你可以在自己的项目中增加一个buildSrc的module来专门处理一些构建时要用到的自定过程,可以使用groovy或者Java来编写。
本着对新事物探索的热爱,我也是一边学gradle一边试着用。从最开始在build.gradle中进行dex指定分包,到现在的可以配置化的Dex分包的gradle插件,也是走过一条长路。
===================================
更新:
1.5.9: 兼容一些古老的 gradle 和 android gradle plugin版本
1.5.8: 兼容gradle 3.2,修复当只使用support-split/support-keep时出现大量的额外类
1.5.7: 修复support-split/support-keep无效的bug
1.5.6:支持java 1.7,修复但只有keep选项时没有类被选中
1.5.5: 单独的maindexlist过滤设置
1.5.5.alpha: 实验性的支持 2.2.0 plugin
1.5.4: instant run 模式时自动禁用DexKnife
1.5.3: 增加跟踪日志,并在jarMerging为null时跳过处理
1.5.2: 修复include和exclude, 并支持过滤单个类(使用'.class'作为单个类的后缀)
1.5.1.exp 版本可以支持2.0以上的plugin
==============================
注意:DexKnife 本身只适用于将配置的通配符生成maindexlist,无法进行多个dex指定。
言归正传,指定dex分包插件为gradle-dexknife-plugin,使用他只需要做几件事:
1、com.android.tools.build:gradle 的版本可以是大于1.3.0的任何版本。
2.0.0以上的instant-run 与 multidex不兼容。在instant-run模式下,DexKnife会暂时自动禁用,在release打包时会自动启用。instant-run 采用新的打包模式,不会出现数量溢出的问题。
2、在项目的build.gradle 增加依赖,我发把他发布到jcenter上了(先检查您的网络是否可达):
buildscript {
...
dependencies {
....
classpath 'com.ceabie.dextools:gradle-dexknife-plugin:1.5.9'
}
}
注意,请确保使用的gradle版本和android gradle plugin兼容,否则会出现同步错误,例如:
Gradle sync failed: Unable to load class 'com.android.builder.core.EvaluationErrorReporter'.
3、在App的module的目录下增加dexknife.txt文件,并编辑你需要的配置
使用 # 进行注释, 当行起始加上 #, 这行配置被禁用.
# 全局过滤, 如果没设置 -filter-suggest 并不会应用到 建议的maindexlist.
# 如果你想要某个包路径在maindex中,则使用 -keep 选项,即使他已经在分包的路径中.
-keep android.support.v4.view.**
# 这条配置可以指定这个包下类在第二dex中.
android.support.v?.**
# 使用.class后缀,代表单个类.
-keep android.support.v7.app.AppCompatDialogFragment.class
# 不包含Android gradle 插件自动生成的miandex列表.
-donot-use-suggest
# 将 全局过滤配置应用到 建议的maindexlist中, 但 -donot-use-suggest 要关闭.
-filter-suggest
# 不进行dex分包, 直到 dex 的id数量超过 65536.
-auto-maindex
# dex 扩展参数, 例如 --set-max-idx-number=50000
# 如果出现 DexException: Too many classes in --main-dex-list, main dex capacity exceeded,则需要调大数值
-dex-param --set-max-idx-number=50000
# 显示miandex的日志.
-log-mainlist
# 如果你只想过滤 建议的maindexlist, 使用 -suggest-split 和 -suggest-keep.
# 如果同时启用 -filter-suggest, 全局过滤会合并到它们中.
-suggest-split **.MainActivity2.class
-suggest-keep android.support.multidex.**
类路径的通配符与gradle其他用到的exclude 和include类似:
'*' 匹配任意数量的字符
'?' 匹配任意一个字符
'**' 匹配任何类路径和类
'.' 和 '/' 都可用来指定类的分割.
'.' 或者 '/' 的结尾会自动增加 '**'.
例如,android.*.v?.** ,android.support.v7 和v4下的所有类都被分到的第二个dex中。
注意: 如果你要过滤内部类, 使用$*,例如: SomeClass$*.class
4、在你的App的module中的build.gradle中加插件:
apply plugin: 'com.ceabie.dexnkife'
注意:要在 defaultConfig 或者 buildTypes中打开 multiDexEnabled true,否则不起作用。
最后,运行的app,分包的东西就交给DexKnife了。祝好运。
(dexknife只会在配置变动或者代码变动的时候才会处理)