android 傻瓜式 MultiDex 插件,从此再也不用担心方法数问题!

ndroid-Easy-MultiDex

项目地址:TangXiaoLv/Android-Easy-MultiDex

简介:Android 傻瓜式 MultiDex 插件,从此再也不用担心方法数问题!

注 1:不想看前半部分的话可以直接跳过到最下面配置部分。
注 2:本插件是基于DexKnifePlugin 1.5.6优化改造而来,感谢 ceabie 的无私奉献。

填坑之路

坑 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 有这问题。

时间: 2024-08-24 06:01:30

android 傻瓜式 MultiDex 插件,从此再也不用担心方法数问题!的相关文章

吃透【预解释】,从此再也不用担心!

Author:李金涛 Form:光环国际 Time:2017-12-31 23:49(跨年夜的最后一刻,我在辛勤耕耘我的"预解释",收获满满,甚喜!) 定义:预解释(变量提升):js在运行前,先把所有带var和function关键字的提前声明或定义.且预解释是发生当前作用域下的. 1,全局预解释阶段: (1)全局作用域与全局变量:当浏览器加载HTML页面的时候,首先会提供一个供全局JavaScript代码执行的环境,称之为全局作用域(global/ window).在window全局作

Swift详解之四-------妈妈再也不用担心我的闭包了

妈妈再也不用担心我的闭包了 注:本文为作者自己总结,过于基础的就不再赘述 ,都是亲自测试的结果.如有错误或者遗漏的地方,欢迎指正,一起学习. swift中闭包是一个很强大的东西,闭包是自包含的函数代码块,可以在代码中被传递和使用.跟C 和 Objective-C 中的代码块(blocks)很相似 .这个大家必须掌握!必须掌握!必须掌握!重要的事情要说三遍 闭包可以捕获和存储其所在上下文中任意常量和变量的引用. 这就是所谓的闭合并包裹着这些常量和变量,俗称闭包.下面我们就来攻克它! 1.闭包函数

服务器要高防就选【韩国KA高防机房】多台金盾万兆NP防火墙集群,免备案,再也不用担心被打死!

在这个恶意竞争的社会,你还在担心服务器被攻击吗?还在担心服务器被打死吗?告诉你使用韩国KA高防服务器,再也不用担心服务器被打死了.韩国KA服务器可提供单防100G(100G UDP+10G TCP),多台金盾万兆NP企业级硬件防火墙集群,单机承诺10G/30G硬防,超出防御自动屏蔽IP,攻击过去立即解封. 韩国KA机房: 单机10G防御区 至强E3-1230 4G  500G 10Mbps 独立IP  月付1699元 至强E3-1230 8G  500G 10Mbps 独立IP  月付1999元

完全免费,再也不用担心转pdf文件乱来乱去的问题了

完全免费,再也不用担心转pdf文件乱来乱去的问题了. 源代码:https://github.com/xlgwr/WpsToPdf.git 第三方插件Bye Bye... 功能说明 主要引用Wps金山办公软件生成pdf功能 可部署到IIS服务器,直接调用API,生成PDF文件 可转文件类型,xls,xlsx,ppt,pptx,doc,docx等Wps金山办公软件所有支持的文件类型. 安装最新Wps 金山办公软件 引用Wps 对应com dll 使用VS2019 打开,生成,运行 web api 发

妈妈再也不用担心我找不到文件了---find

1.find vs locate 在实际中,我们经常需要查找到一些特定文件,然后进行处理,LINUX提供了locate , find这两个命令用于文件查找. a.locate,非实时查找,非精确查找.linux会定期生成更新文件数据库,而locate将根据文件数据库进行查找.我们可以在使用locate命令前,更新文件数据库,使用updatedb即可.但是updatedb将会花费可能半天时间,SO LONG! b.find , 实时查找,精确查找.根据指定路径,查找标准,进行文件遍历(包括隐藏文件

利用CH341A编程器刷新BIOS,恢复BIOS,妈妈再也不用担心BIOS刷坏了

前几天,修电脑主析就捣鼓刷BIOS,结果刷完黑屏开不了机,立刻意识到完了,BIOS刷错了.就从网上查资料,各种方法试了个遍,什么用处都没有.终于功夫不负有心人,找到了编码器,知道了怎么用.下面看看具体用法: 先买了一个便宜点的编码器: <ignore_js_op> 把刷错的BIOS芯片先拆下来:<ignore_js_op> 把芯片放到编码器上: 再找一台电脑,我的笔记本派上用场了,下载编码器的驱动程序和编码器软件:  <ignore_js_op> <ignore_

Xcode7的发布后的crash跟踪,轻松定位崩溃代码 Address Sanitizer: 妈妈再也不用担心 EXC_BAD_ACCESS

Xcode7中苹果为我们增加了两个重要的debug相关功能.了解之后觉得非常实用,介绍给大家. 1.Address Sanitizer: 妈妈再也不用担心 EXC_BAD_ACCESS? EXC_BAD_ACCESS一直是很多开发者的噩梦,因为这个错误很不直观,出现后往往要花很长时间才能定位到错误.苹果这次带来了革命性的提升. 在项目的Scheme中Diagnostics下,选中enable address sanitizer(注意选中后Xcode会重新编译整个项目). 这样设置后,如果再出现类

妈妈再也不用担心我js跨域了。。

$.getJSON('http://www.example.com/?aa=bb&callback=?', function(r){ console.log(r) }); //callback=?参数带上,妈妈再也不用担心我跨域了.jquery的getJSON利用的也是JSONP的原理去实现的.

keycode 大全,javascript 再也不用担心我不知道的keycode了

keycode    8 = BackSpace BackSpace keycode    9 = Tab Tab keycode   12 = Clear keycode   13 = Enter keycode   16 = Shift_L keycode   17 = Control_L keycode   18 = Alt_L keycode   19 = Pause keycode   20 = Caps_Lock keycode   27 = Escape Escape keycod