Android插件化的目的主要有两个,第一个是应对每个dex包65536个方法数的上限问题,第二是实现功能复杂的app的拆解,能够按需下载和加载运行所需的模块。插件化的实现并没有统一的标准或方式,很多公司都有自己的一套插件化方案,而且没有哪个公司的方案被业界普遍认可。本文将按时间轴的方式介绍几家比较有代表性的公司的插件化方案,供大家开阔思路。至于要选取那种方案为我所用,那就只有结合自己的实际需求实践比较得知。
Facebook是进行插件化开发尝试的鼻祖,早期Facebook安装包变大以后出现的两个导致安装失败的问题:
第一个错误是说方法数目超过最大数目64K,这是因为Android每个Dex文件的方法通过两个字节进行索引,两个字节能索引的最大数目就是64K。第二个错误是说单个dex包大小超出了Dexopt所能使用的最大缓存。Dexopt是将.dex文件进行优化,并转化成.odex的工具,所能使用的最大缓存跟手机相关,一般为8M或16M,如果dex包太大就有可能超出Dexopt所能使用的最大缓存,导致出错。
对此Facebook采取了两种方法来解决以上问题,总结如下:
后来,Google在Android里引入这种问题一的解决方法,MultiDex方法,并将其标准化(5.0以下版本,buildtools升级到21即可使用支持包)。使用步骤:
首先修改gradle文件,在defaultConfig里面添加multiDexEnabled属性,并修改dependencies:
android {
...
defaultConfig {
...
minSdkVersion 14
targetSdkVersion 21
...
// Enabling multidex support.
multiDexEnabled true
}
...
}
dependencies {
compile 'com.android.support:multidex:1.0.0'
}
然后在Application的扩展类中重写attachBaseContext函数:
public class MyApplication extends Application {
//...
@Override
protected void attachBaseContext(Context base) {
super.attachBaseContext(base);
try {
// to resolve: java.lang.NoClassDefFoundError
MultiDex.install(this);
} catch (Exception e) {
e.printStackTrace();
}
}
//...
}
该方法大部分情况下可行,但是存在几个问题:
有些技巧可以缓解存在的上述问题:
美团网
美团网对拆分dex包的过程进行了干预,并针对ANR错误进行了优化,他们要解决的三个问题以及采取的方案如下:
如何控制生成多个dex包 : 在gradle中定义一个task,控制生成多个dex包
如何控制哪些class放到主dex包,哪些放在辅dex包
如何避免动态加载辅dex包导致的ANR错误
微信
微信团队也总结了一套自己的优化方案,他们针对动态加载辅dex包进行了自己的处理,微信主要修改了动态加载辅dex包的方式:
第一次加载辅dex包时需要进行dexopt操作,这个过程比较耗时,但是如果已经加载过一次再加载就会很快了,所以微信的策略是,第一次加载的时候在sharedpreference中写个标记,下次启动如果发现已经有该标记,直接往下走,否则在新开一个透明的Activity,这样这个Activity就是前台进程,主进程就不再是前台进程,也就不会出现ANR错误了。流程如下:
微信动态加载Dex文件流程
携程网
携程网为了实现插件化并行开发,借鉴前辈们的思路开发了一套自己的插件化方案DynamicApk,他们不仅实现了dex包的拆分和动态加载,更重要的是,他们实现了多个插件apk独立并行开发,以便多人协作开发。
面临的问题及解决思路:
实现思路:
实现方式:
上面介绍的几家公司的插件化技术可以分为两类:多dex方案和多apk方案。这两种方案异同总结如下:
两种方案如何选择:
其他技术方案 360的 DroidPlugin开源框架
Xposed
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8