- Java源码编译:通过javac将源码编译为.class文件
- 多DEX分包:脚本将类根据一定规则划分到主dex和从dex中,生成配置文件
- proguard优化混淆:对.class文件进行压缩、优化、混淆处理
- 转换为dex文件:dx\d8将.class文件转换为dex文件
Android dex编译流程
DEX格式结构图
详情了解:一篇文章带你搞懂DEX文件的结构
MutiDex方案背景
64k应用限制: Conversion to Dalvik format failed: Unable to execute dex:method ID not in[0,0xffff]:65536
原因:
1、DexOpt优化的限制
早期Android系统中DexOpt会把每一个类的方法id检索起来,存在一个链表里面。链表的长度是用short类型来保存,导致来方法id的数目不能超过65536个。
2、dalvik bytecode的限制
Dalvik的invoke-kind指令集,设置16bit表示方法引用数,最大值为65536,invoke-kind{vC,vD,vE,vF,vG},meth@BBBB
MutiDex方案引入
Android 5.0以前:使用Dalvik可执行文件分包支持库
Android 5.0以后:Android 5.0及更高版本使用ART运行时,后者原生支持从APK文件加载多个DEX文件。ART在应用安装时执行预编译,扫描classesN.dex文件,并将他们编译成单个.oat文件,供Android设备执行。因此,如果minSdkVersion为21或更高值,则不需要Dalvik可执行文件分包支持库。
综上所述:要不把minSdkVersion为21或更高值,要不就添加分包支持库
修改build.gradle
minSdkVersion为21
android {
defaultConfig {
......
minSdkVersion 21
targetSdkVersion 26
multiDexEnabled true
versionCode 1
versionName "1.0"
......
}
}
添加分包支持库
android {
defaultConfig {
......
minSdkVersion 15
targetSdkVersion 26
multiDexEnabled true
versionCode 1
versionName "1.0"
......
}
dependencies {
......
compile'com.android.support:multidex:1.0.3'
}
}
修改Application代码(两种方式)
1、Application继承MultiDexApplication
2、调用MultiDex.install(this)启用分包
MutiDex分包流程
- 生成manifest_keep.txt
解析出AndroidManifest.xml中所有的组件类:包括Activity、Service、Receiver以及ContentProvider,这些类将和Application入口类一起放在manifest_keep.txt文件中。 - 生成maindexlist.txt文件
查找mainfest_keep.txt中所有类的直接引用类,将其放保存在maindexlist.txt - 生成多个dex
dx工具接收参数,将maindexlist.txt文件包含的所有class编译进主dex,并生成其他dex。
MutiDex加载原理
BaseDexClassLoader(findClass)
|
DexPathList (遍历dexElements)