问题1:65536 ,So easy!

原因:Dalvik 的 invoke-kind 指令集中,method reference index 只留了 16 bits,最多能引用 65535 个方法。
参考=>由 Android 65K 方法数限制引发的思考.

解决:

dependencies { compile 'com.android.support:MultiDex:1.0.1'
}

继承 Application ,重写 attachBaseContext(Context)

@Override
protected void attachBaseContext(Context base) {super.attachBaseContext(base);MultiDex.install(this);
}

问题2:Too many classes in –main-dex-list ,what?

原因:通过上面的官方分包,已经把原 Dex 分为 1 主 Dex 加多从 Dex。主 Dex 包含所有 4 大组件,Application,Annotation,multidex 等及其必要的直接依赖。由于我们方法数已达到 16W 之巨,上百个 Activity 全部塞进主 Dex,又成功的把主 Dex 撑爆了。

解决: gradle

afterEvaluate { tasks.matching { it.name.startsWith('dex') }.each { dx -> if (dx.additionalParameters == null) { dx.additionalParameters = []}  dx.additionalParameters += '--set-max-idx-number=48000' }
}

参考=>Android Dex 分包之旅

问题3:gradle 1.5.0 之后不支持这种写法 ,what the fuck?

原因:官方解释 Gralde1.5.0以上已经将(jacoco, progard, multi-dex)统一移到Transform API里,然而 Transform API 并没有想象的那么简单好用,翻遍 Google 终于找到一个兼容 Gradle 1.5.0以上的分包插件DexKnifePlugin。
扩展=>这篇Android 热修复使用 Gradle Plugin1.5 改造 Nuwa 插件比较好的介绍了 Transform API 的使用。

问题4:NoClassDefFoundError ,are you kiding me?

原因:通过插件手动指定 main dex 中要保留的类,虽然分包成功,但是 main dex 中的类及其直接引用类很难通过手动的方式指定。

解决方式:
美团 Android DEX 自动拆包及动态加载简介,他们是通过编写了一个能够自动分析 Class 依赖的脚本去算出主 Dex 需要包含的所有必要依赖。看来写脚本是跑不掉了。

问题5:自定义脚本 ,read the fuck source!

问题一:哪些类是需要放入主 Dex 中?
查看 sdk\build-tools\platform-version\mainDexClasses.rules 发现放入主 Dex 相关类有 Instrumentation,Application,Activity,Service,ContentProvider,BroadcastReceiver,BackupAgent 的所有子类。

问题二:gradle 是在哪里算出主 Dex 依赖?
查看 Gradle 编译任务发现有如下 3 个编译任务:

运行 collect 任务,发现会在 build/multi-dex 目录下单独生成manifest_keep.txt文件,该文件其实就是通过上述规则扫描AndroidManifest生成。manifest_keep.txt保留的是所有需要放入主 Dex 里的类。还没完,接下来transformClassesWithMultidexlist任务会根据manifest_keep.txt生成必要依赖列表maindexlist.txt,这里面所有类才是真正放入主 Dex 里的。bingo,现在非常清楚,我们只需要控制进入 manifest_keep.txt 中的类即可,最终其类的依赖关系由系统帮我们生成即可,安全绿色可靠!


问题三:在哪里控制maindexlist.txt的大小?
由问题一我们知道生成manifest_keep.txt的规则,对于绝大部分工程来说,manifest_keep.txt中 80%是 Activity,其实我们并不需要把全部的 Activity 放入主 Dex,只需要保留必要的 Activity 即可,如首页 Activity、Laucher Activity 、欢迎页的 Activity 等启动时必要的 Activity 就 OK 了。

下图是 Gradle 的工作流程: 
来源:深入理解 Android 之 Gradle

我们只需要在完成任务向量图之后,执行任务之前 Hook 一下 collect 任务,过滤掉不必要的 activity 就 OK 了。添加 Gradle:

//需要加入主 dex 的 Activity 列表
def mainDexListActivity = ['WelcomeActivity', 'MainFunctionActivity']
afterEvaluate {project.tasks.each { task ->if (task.name.startsWith('collect') && task.name.endsWith('MultiDexComponents')) {println "main-dex-filter: found task $task.name"task.filter { name, attrs ->String componentName = attrs.get('android:name')if ('activity'.equals(name)) {def result = mainDexListActivity.find {componentName.endsWith("${it}")}return result != null} else {return true}}}}
}

问题6:主 dex 依然爆表,shit again!

其实上面那段脚本已经成功筛选出我们想要放入主 Dex 的manifest_keep 列表maindexlist 列表,但是在打包的时候还是把所有类打进主 Dex(已无语)。这个时候就需要跟DexKnifePlugin插件配合使用,首先在 gradle 中加上上述脚本,然后使用插件时在配置文件中加上 -split **.**#-donot-use-suggest。DexKnifePlugin 插件运行原理很简单,在生成 Dex 任务之前首先读取自己的配置文件(包含前面我们通过 Gradle 脚本生成的maindexlist列表),然后扫描 combined.jar(包含工程中所有.class 文件)匹配出我们自定义的 maindexlist.txt,再替换掉 build/multi-dex/maindexlist.txt,和 build 实例。这样分包的时候就会基于我们的规则生成主 Dex。

问题7:ANR,HAHAHA!

我们最低 API=16,测试并未发现 ANR 问题,所以暂时没考虑景上添花,这个问题比较好解决。
参考=>Android Dex 分包之旅

Congratulation

恭喜,填坑终于结束,不过还有点不爽的是需要同时维护 Gradle 脚本和插件的配置。 于是乎就将 Gradle 脚本整合进了插件,这样只需维护一个配置文件就行了。读者可以根据自己需求自行选择分开配置还是整合配置。通过这种方式我们把主 Dex 的方法数维持在 15000 左右,从此再也不用担心方法数问题了!!!

配置部分

第一步:添加根目录 Gradle

buildscript {dependencies {classpath 'com.library.tangxiaolv:dexknife-plus:1.0.1'}
}

第二步:在你的 App 模块的 build.gradle 添加插件

apply plugin: 'dexknifePlus'

第三步:配置参数

dexKnife{//必选参数enabled true //if false,禁用分包插件//可选参数//1.如果没有可选参数,将根据 enabled 决定是否分包。//2.如果有可选参数,需满足必选参数和可选参数的条件才允许分包productFlavor 'mock'buildType 'debug'/**eg:当前 productFlavors = dev,buildType = debug,*参数组合 1:enabled = true,productFlavor = dev,buildType = debug 分包*参数组合 2:enabled = true,productFlavor = mock,buildType = debug 不分包*参数组合 1:enabled = true,buildType = debug 所有 buildType = debug 分包*参数组合 1:enabled = true,productFlavor = dev 所有 productFlavor = dev 分包* */
}

第四步:在你的 App 模块目录下新建 dexknife.txt,并自定义配置

#为注释符#-----------主 Dex 中必要依赖的脚本配置-----------
#默认保留四大组件中其他三大组件(并计算其依赖树),Activity 组件选择性保留(使用-just activity 选项),若为空不保留任何 Activity
-just activity com.ceabie.demo.MainActivity#-----------附加类-----------
# 如果你想要某个包路径在 maindex 中,则使用 -keep 选项,即使他已经在分包的路径中.若为空,不保留任意类
#-keep com.ceabie.demo.**# 保留单个类.
#-keep android.support.v7.app.AppCompatDialogFragment.class# 这条配置可以指定这个包下类在第二及其他 dex 中.
#-split android.support.v?.**
#将全部类移出主 Dex
-split **.**# 不包含 Android gradle 插件自动生成的 miandex 列表.(不使用建议的依赖树,注释掉表示使用,否则-just activity 无效)
#-donot-use-suggest# (分割每个 dex 包的方法数上限) 扩展参数:例如 --set-max-idx-number=50000
# 如果出现 DexException: Too many classes in --main-dex-list, main dex capacity exceeded:
# 表明限制的方法数小于 main dex 的必要方法数,调大到合适数值即可
-dex-param --set-max-idx-number=4000# 不进行 dex 分包, 直到 dex 的 id 数量超过 65536.(设置自动执行分包策略)
#-auto-maindex# 显示 miandex 的日志.
#-log-mainlist

第五步:在 defaultConfig 或者 buildTypes 中打开 multiDexEnabled true,否则不起作用

已知错误

注:分包的时候如果发现一些莫名的错误,可以关掉 instant run,一般都能解决

错误 1: (已修复)

Error:Execution failed for task ':Toon:transformClassesWithDexForDebug'.> java.lang.NullPointerException (no error message)

发生此错误只要切换一次 Gradle 版本就 OK 了,比如 1.5.0

错误 2:(已修复)

Unsupported major.minor version 52.0

由于插件中使用到了 JDK1.8 的一些 API,所以将 JDK 升级到 1.8 就可以了

错误 3: (已修复)

Error:Execution failed for task ':app:transformClassesWithDexForDebug'.
> DexKnife Warnning: Main dex is EMPTY ! Check your config and project!

gradle 切到 1.5.0,目前就发现 gradle 2.1.2 有这问题。

有其他问题或者技术困惑的伙伴,可以加群交流(备注技术交流)

MultiDex 相关问题解决记录相关推荐

  1. 华为交换机密码相关问题解决记录

    重置Console密码 一 重启交换机 直接重启交换机,如果没有开关键,直接拔电源重启. (温馨提醒:这种操作仅适合未上线的设备,没有跑业务的交换机进行操作.如果是线上的交换机需要进行破解密码,请根据 ...

  2. Hadoop相关问题解决--记录

    Hive 1.查询hivemeta信息,查到的numRows为-1 cdh 不限 不限 不限 在hivemeta库中可以通过以下sql查询表的元数据信息 SELECT * FROM TABLE_PAR ...

  3. ADOBE CS3 安装相关问题解决

    ADOBE CS3 安装相关问题解决(部分解决方案收集自互联网,没有经过测试) ps:2008-4-26 补充一下,有时cs3安装不成功,是因为装到了中文目录下了.应该装到英文目录下,这一点也需要注意 ...

  4. dolphinscheduler 出现 “tenant not exists” 租户不存在的相关问题解决

    dolphinscheduler 出现 "tenant not exists" 租户不存在的相关问题解决 1. 问题记录 dolphinscheduler2.0.5在执行datax ...

  5. [企业化NET]Window Server 2008 R2[1]-服务器基本安装即问题解决记录

    1.  服务器基本安装即问题解决记录      √ 2.  SVN环境搭建和客户端使用 2.1  服务端 和 客户端 安装    √ 2.2  项目建立与基本使用     √ 2.3  基本冲突解决, ...

  6. iOS 10、Xcode 8 遇到部分问题解决记录(包括控制台日志不输出)

    iOS 10.Xcode 8 遇到部分问题解决记录(包括控制台日志不输出) 参考文章: (1)iOS 10.Xcode 8 遇到部分问题解决记录(包括控制台日志不输出) (2)https://www. ...

  7. MySQL安装步骤及相关问题解决

    MySQL安装步骤及相关问题解决 参考文章: (1)MySQL安装步骤及相关问题解决 (2)https://www.cnblogs.com/baimt/p/5688517.html 备忘一下.

  8. HTTP请求状态码404相关问题解决

    HTTP请求状态码404相关问题解决 参考文章: (1)HTTP请求状态码404相关问题解决 (2)https://www.cnblogs.com/complc/p/11646710.html 备忘一 ...

  9. Google Maps API v2 android版本开发 国内手机不支持google play Service相关问题解决--图文教程

    Google Maps API v2 android版本开发 国内手机不支持google play Service相关问题解决--图文教程 参考文章: (1)Google Maps API v2 an ...

最新文章

  1. 一学就会的 Python 时间转化总结(超全)
  2. 一些惹起热烈争议的PCB布线经验法则
  3. Cordova error: tool 'xcodebuild' requires Xcode, but active developer directory '/Library/Deve
  4. EnforceLearning-在线学习-被动强化学习/评价学习
  5. 不改代码也能全面 Serverless 化,阿里中间件如何破解这一难题?
  6. python scrapy框架基如何实现多线程_Python实现在线程里运行scrapy的方法
  7. javascript表单处理相关的知识总结(一)
  8. 【华为云技术分享】技术干货丨通过wrap malloc定位C/C++的内存泄漏问题
  9. Java基本数据类型及其包装类
  10. 【机器学习】MATLAB读取mnist数据库
  11. Qt--音乐播放器 V2.0
  12. 54. C# -- 泛型(Generic)
  13. 对vector/string执行insert/erase操作后迭代器的情况说明
  14. 如何将PDF转换成可以直接编辑的CAD图纸
  15. c语言数组的地址传递,c语言函数传递数组_c语言函数数组地址传递没有输出
  16. 纳米盘资源搜索经验分享
  17. 允许用户使用 MAK 密钥激活 Office 2010 批量许可版
  18. oracle remap schema,impdp的remap_schema选项的另一个schema是否要重建
  19. 安徽省大数据与人工智能竞赛经验分享-3【从赛题中分析比赛需要的技能】
  20. 传统企业如何做数字化转型?弄懂这3大底层逻辑你就懂了

热门文章

  1. 集成学习【一】:集成学习(Ensemble Learning)结合神经网络
  2. Windows Server 2012 R2 部署JavaWeb项目之环境、软件、配置
  3. cad完全卸载教程_如何完全卸载(删除)cad吗? _ 设计学院_设计软件教程自学网...
  4. python爬虫之路【2】fiddle手机抓包
  5. 小马2K3PE永久珍藏版|
  6. 算法:每个元音包含偶数次的最长字符串
  7. RNA-seq数据下载
  8. 机器人课程一切都在快速变化但又好像什么都没变^_^
  9. 30天自制操作系统_序
  10. 直观理解-梯度下降及MIT自适应控制律