Android分包MultiDex策略详解

作者:灰色飘零 时间:2021-11-14 13:28:42 

1.分包背景

这里首先介绍下MultiDex的产生背景。

当Android系统安装一个应用的时候,有一步是对Dex进行优化,这个过程有一个专门的工具来处理,叫DexOpt。DexOpt的执行过程是在第一次加载Dex文件的时候执行的。这个过程会生成一个ODEX文件,即Optimised Dex。执行ODex的效率会比直接执行Dex文件的效率要高很多。

但是在早期的Android系统中,DexOpt有一个问题,DexOpt会把每一个类的方法id检索起来,存在一个链表结构里面。但是这个链表的长度是用一个short类型来保存的,导致了方法id的数目不能够超过65536个。当一个项目足够大的时候,显然这个方法数的上限是不够的。尽管在新版本的Android系统中,DexOpt修复了这个问题,但是我们仍然需要对低版本的Android系统做兼容。

为了解决方法数超限的问题,需要将该dex文件拆成两个或多个,为此谷歌官方推出了multidex兼容包,配合AndroidStudio实现了一个APK包含多个dex的功能。

我们在Android开发中,会不断的在App代码里面增加新功能,引入新的类库,如果不加控制的话,那么会碰到编辑器IDE爆出一下错误:


Error:Execution failed for task ':ttt:transformClassesWithDexForDebug'.
com.android.build.api.transform.TransformException: com.android.ide.common.process.ProcessException: java.util.concurrent.ExecutionException: com.android.dex.DexIndexOverflowException: method ID not in [0, 0xffff]: 65536

这个错误是Android应用的对方法总数有限制造成的。Android平台的Java虚拟机Dalvik在执行DEX格式的Java应用程序时,使用原生类型short来索引DEX文件中的方法。这意味着单个DEX文件可被引用的方法总数被限制为65536。通常APK包含一个classes.dex文件,因此Android应用的方法总数不能超过这个数量,这包括Android框架、类库和你自己开发的代码。

这个问题可以通过将一个DEX文件分拆成多个DEX文件解决。

2. 分包策略实现

Gradle 配置:


defaultConfig {  
   applicationId "XXX"
   minSdkVersion 14
   targetSdkVersion 23
   multiDexEnabled true
}
.......
dependencies {
 compile 'com.android.support:multidex:1.0.0'
}

在应用的Application 类重写方法:


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

3.分包效果说明

经过以上的配置,你的应用已经可以实现多个DEX文件了。当应用构建时,构建工具会分析哪些类必须放在第一个DEX文件,哪些类可以放在附加的DEX文件中。当它创建了第一个DEX文件后,如果有必要会继续创建附加的DEX文件,如classes2.dex, classes3.dex。Multidex的支持类库将被包含在应用的第一个DEX文件中,帮助实现对其它DEX文件的访问。

虽然Google解决了应用总方法数限制的问题,但并不意味着开发者可以任意扩大项目规模。Multidex仍有一些限制:

DEX文件安装到设备的过程非常复杂,如果第二个DEX文件太大,可能导致应用无响应。此时应该使用ProGuard减小DEX文件的大小。

由于Dalvik linearAlloc的Bug,应用可能无法在Android 4.0之前的版本启动,如果你的应用要支持这些版本就要多执行测试。
同样因为Dalvik linearAlloc的限制,如果请求大量内存可能导致崩溃。Dalvik linearAlloc是一个固定大小的缓冲区。在应用的安装过程中,系统会运行一个名为dexopt的程序为该应用在当前机型中运行做准备。dexopt使用LinearAlloc来存储应用的方法信息。Android 2.2和2.3的缓冲区只有5MB,Android 4.x提高到了8MB或16MB。当方法数量过多导致超出缓冲区大小时,会造成dexopt崩溃。

-Multidex构建工具还不支持指定哪些类必须包含在首个DEX文件中,因此可能会导致某些类库(例如某个类库需要从原生代码访问Java代码)无法使用。

4.对开发者的建议

开发者应该避免使用Google Guava这样的类库,它包含了13000多个方法。

尽量使用专为移动应用设计的Lite/Android版本类库,或者使用小类库替换大类库,例如用Google-gson替换Jackson JSON。而对于Google Protocol Buffers这样的数据交换格式,其标准实现会自动生成大量的方法。采用Square Wire的实现则可以很好地解决此问题。

在出现应用分包后低版本手机无法使用,高版本正常使用的问题时,可以考虑检查一下分包的配置是否正确。

来源:http://www.cnblogs.com/renhui/p/7738421.html

标签:android,multidex
0
投稿

猜你喜欢

  • Linux环境下安装MySQL8.0的完整步骤

    2024-01-22 05:26:47
  • Go语言包管理模式示例分析

    2024-05-22 10:20:17
  • Python删除字符串中字符的四种方法示例代码

    2021-09-25 08:23:25
  • python脚本框架webpy入门安装及应用创建

    2023-08-23 05:08:51
  • 解决vue组件中click事件失效的问题

    2023-07-02 16:34:10
  • pycharm打包py项目为.exe可执行文件的两种方式

    2022-03-26 11:51:51
  • 教你两步解决conda安装pytorch时下载速度慢or超时的问题

    2022-04-12 17:19:59
  • webpack动态加载与打包方式

    2024-04-23 09:16:10
  • 网页设计十大诀窍

    2007-10-19 13:03:00
  • 解决Tensorflow安装成功,但在导入时报错的问题

    2022-03-05 18:39:27
  • SQL2005 高效分页sql语句

    2024-01-17 13:23:22
  • python模块常用用法实例详解

    2023-07-30 01:38:54
  • Python通过matplotlib画双层饼图及环形图简单示例

    2021-09-14 11:08:18
  • [精品]ASP中常用的22个FSO文件操作函数

    2007-08-18 15:12:00
  • python基于pdfminer库提取pdf文字代码实例

    2023-11-08 06:31:22
  • 基于python实现文件加密功能

    2022-02-02 20:28:57
  • Insert into与AddNew哪一个更好?

    2009-10-28 18:30:00
  • python 按照固定长度分割字符串的方法小结

    2021-05-28 06:05:05
  • Python Scapy随心所欲研究TCP协议栈

    2023-06-10 23:12:29
  • python3 求约数的实例

    2023-12-29 03:18:59
  • asp之家 网络编程 m.aspxhome.com