原文地址: http://my.oschina.net/853294317/blog/308583

当一个app的功能越来越复杂,代码量越来越多,也许有一天便会突然遇到下列现象:

1. 生成的apk在2.3以前的机器无法安装,提示INSTALL_FAILED_DEXOPT

2. 方法数量过多,编译时出错,提示:

Conversion to Dalvik format failed:Unable to execute dex: method ID not in [0, 0xffff]: 65536

出现这种问题的原因是:

1. Android2.3及以前版本用来执行dexopt(用于优化dex文件)的内存只分配了5M

2. 一个dex文件最多只支持65536个方法。

针对上述问题,也出现了诸多解决方案,使用的最多的是插件化,即将一些独立的功能做成一个单独的apk,当打开的时候使用DexClassLoader动态加载,然后使用反射机制来调用插件中的类和方法。这固然是一种解决问题的方案:但这种方案存在着以下两个问题:

1. 插件化只适合一些比较独立的模块;

2. 必须通过反射机制去调用插件的类和方法,因此,必须搭配一套插件框架来配合使用;

由于上述问题的存在,通过不断研究,便有了dex分包的解决方案。简单来说,其原理是将编译好的class文件拆分打包成两个dex,绕过dex方法数量的限制以及安装时的检查,在运行时再动态加载第二个dex文件中。faceBook曾经遇到相似的问题,具体可参考:

https://www.facebook.com/notes/facebook-engineering/under-the-hood-dalvik-patch-for-facebook-for-android/10151345597798920

文中有这么一段话:

However, there was no way we could break our app up this way--too many of our classes are accessed directly by the Android framework. Instead, we needed to inject our secondary dex files directly into the system class loader。

文中说得比较简单,我们来完善一下该方案:除了第一个dex文件(即正常apk包唯一包含的Dex文件),其它dex文件都以资源的方式放在安装包中,并在Application的onCreate回调中被注入到系统的ClassLoader。因此,对于那些在注入之前已经引用到的类(以及它们所在的jar),必须放入第一个Dex文件中。

下面通过一个简单的demo来讲述dex分包方案,该方案分为两步执行:

整个demo的目录结构是这样,我打算将SecondActivity,MyContainer以及DropDownView放入第二个dex包中,其它保留在第一个dex包。

一、编译时分包

整个编译流程如下:

除了框出来的两Target,其它都是编译的标准流程。而这两个Target正是我们的分包操作。首先来看看spliteClasses target。

由于我们这里仅仅是一个demo,因此放到第二个包中的文件很少,就是上面提到的三个文件。分好包之后就要开始生成dex文件,首先打包第一个dex文件:

由这里将${classes}(该文件夹下都是要打包到第一个dex的文件)打包生成第一个dex。接着生成第二个dex,并将其打包到资资源文件中:

可以看到,此时是将${secclasses}中的文件打包生成dex,并将其加入ap文件(打包的资源文件)中。到此,分包完毕,接下来,便来分析一下如何动态将第二个dex包注入系统的ClassLoader。

二、将dex分包注入ClassLoader

这里谈到注入,就要谈到Android的ClassLoader体系。

由上图可以看出,在叶子节点上,我们能使用到的是DexClassLoader和PathClassLoader,通过查阅开发文档,我们发现他们有如下使用场景:

1. 关于PathClassLoader,文档中写到: Android uses this class for its system class loader and for its application class loader(s),

由此可知,Android应用就是用它来加载;

2. DexClass可以加载apk,jar,及dex文件,但PathClassLoader只能加载已安装到系统中(即/data/app目录下)的apk文件。

知道了两者的使用场景,下面来分析下具体的加载原理,由上图可以看到,两个叶子节点的类都继承BaseDexClassLoader中,而具体的类加载逻辑也在此类中:

BaseDexClassLoader:

[java] view plaincopy

  1. @Override

  2. protected Class<?> findClass(String name) throws ClassNotFoundException {

  3. List<Throwable> suppressedExceptions = new ArrayList<Throwable>();

  4. Class c = pathList.findClass(name, suppressedExceptions);

  5. if (c == null) {

  6. ClassNotFoundException cnfe = new ClassNotFoundException("Didn't find class \"" + name + "\" on path: " + pathList);

  7. for (Throwable t : suppressedExceptions) {

  8. cnfe.addSuppressed(t);

  9. }

  10. throw cnfe;

  11. }

  12. return c;

  13. }

由上述函数可知,当我们需要加载一个class时,实际是从pathList中去需要的,查阅源码,发现pathList是DexPathList类的一个实例。ok,接着去分析DexPathList类中的findClass函数,

DexPathList:

[java] view plaincopy

  1. public Class findClass(String name, List<Throwable> suppressed) {

  2. for (Element element : dexElements) {

  3. DexFile dex = element.dexFile;

  4. if (dex != null) {

  5. Class clazz = dex.loadClassBinaryName(name, definingContext, suppressed);

  6. if (clazz != null) {

  7. return clazz;

  8. }

  9. }

  10. }

  11. if (dexElementsSuppressedExceptions != null) {

  12. suppressed.addAll(Arrays.asList(dexElementsSuppressedExceptions));

  13. }

  14. return null;

  15. }

上述函数的大致逻辑为:遍历一个装在dex文件(每个dex文件实际上是一个DexFile对象)的数组(Element数组,Element是一个内部类),然后依次去加载所需要的class文件,直到找到为止。

看到这里,注入的解决方案也就浮出水面,假如我们将第二个dex文件放入Element数组中,那么在加载第二个dex包中的类时,应该可以直接找到。

带着这个假设,来完善demo。

在我们自定义的BaseApplication的onCreate中,我们执行注入操作:

[java] view plaincopy

  1. public String inject(String libPath) {

  2. boolean hasBaseDexClassLoader = true;

  3. try {

  4. Class.forName("dalvik.system.BaseDexClassLoader");

  5. catch (ClassNotFoundException e) {

  6. hasBaseDexClassLoader = false;

  7. }

  8. if (hasBaseDexClassLoader) {

  9. PathClassLoader pathClassLoader = (PathClassLoader)sApplication.getClassLoader();

  10. DexClassLoader dexClassLoader = new DexClassLoader(libPath, sApplication.getDir("dex", 0).getAbsolutePath(), libPath, sApplication.getClassLoader());

  11. try {

  12. Object dexElements = combineArray(getDexElements(getPathList(pathClassLoader)), getDexElements(getPathList(dexClassLoader)));

  13. Object pathList = getPathList(pathClassLoader);

  14. setField(pathList, pathList.getClass(), "dexElements", dexElements);

  15. return "SUCCESS";

  16. catch (Throwable e) {

  17. e.printStackTrace();

  18. return android.util.Log.getStackTraceString(e);

  19. }

  20. }

  21. return "SUCCESS";

  22. }

这是注入的关键函数,分析一下这个函数:

参数libPath是第二个dex包的文件信息(包含完整路径,我们当初将其打包到了assets目录下),然后将其使用DexClassLoader来加载(这里为什么必须使用DexClassLoader加载,回顾以上的使用场景),然后通过反射获取PathClassLoader中的DexPathList中的Element数组(已加载了第一个dex包,由系统加载),以及DexClassLoader中的DexPathList中的Element数组(刚将第二个dex包加载进去),将两个Element数组合并之后,再将其赋值给PathClassLoader的Element数组,到此,注入完毕。

现在试着启动app,并在TestUrlActivity(在第一个dex包中)中去启动SecondActivity(在第二个dex包中),启动成功。这种方案是可行。

但是使用dex分包方案仍然有几个注意点:

1. 由于第二个dex包是在Application的onCreate中动态注入的,如果dex包过大,会使app的启动速度变慢,因此,在dex分包过程中一定要注意,第二个dex包不宜过大。

2. 由于上述第一点的限制,假如我们的app越来越臃肿和庞大,往往会采取dex分包方案和插件化方案配合使用,将一些非核心独立功能做成插件加载,核心功能再分包加载。

Android dex分包方案 (多dex)相关推荐

  1. dex分包方案概述与multidex包的配置使用

    参考资料: Android dex分包方案 Android分包MultiDex原理 <Android开发艺术探索> 博客中间会涉及到dex文件的反编译,参考博文: dex文件的反编译-de ...

  2. 干货满满,Android热修复方案介绍

    摘要:在云栖社区技术直播中,阿里云客户端工程师李亚洲(毕言)从技术原理层面解析和比较了业界几大热修复方案,揭开了Qxxx方案.Instant Run以及阿里Sophix等热修复方案的神秘面纱,帮助大家 ...

  3. android热修复方案

    热补丁方案有很多,其中比较出名的有腾讯Tinker.阿里的AndFix.美团的Robust以及QZone的超级补丁方案.他们的优劣如下: 一.Tinker 热修复 Tinker通过 Dexdiff 算 ...

  4. Android Multidex(dex分包)

    Android Multidex(dex分包) 分包: 一个dex分成多个dex 什么要分包 单个 Dalvik Executable (DEX) 中, 可调用的最大的引用总数为 65536 ,若超过 ...

  5. Android DEX加固方案与原理

    Android 反编译的威胁 逆向分析: 漏洞挖掘.协议分析 二次打包: 盗版.破解.广告 保护方案 代码混淆:Java代码.C\C++带马甲.JS\HTML代码 应用加固:DEX文件.SO文件.资源 ...

  6. 【Android 热修复】热修复原理 ( 多 Dex 打包机制 | 多 Dex 支持 | Dex 分包设置 | 开发和产品风格设置 | 源码资源 )

    文章目录 一.Dex 打包设置 1.多 Dex 支持 2.Dex 分包设置 3.开发和产品风格设置 ( 非必须 ) 二.完整 build.gradle 配置 1.build.gradle 配置 2.d ...

  7. Android 热修复方案Tinker(三) Dex补丁加载

    转载来源:http://blog.csdn.net/l2show/article/details/53307523 之前有说到Tinker的修复原理是跟Qzone类似,这里就详细分析一下为什么这样做可 ...

  8. Android分包方案multidex

    对于功能越来越复杂的app的两大问题 一:我们自己应用的方法数超过了65536  我们所说的方法数限制,这个方法数包括了jar包,框架,还有我们自己应用的代码,当我们应用的代码超过65536时,结果如 ...

  9. dex分包之--------multidex包的配置使用

    目录: 一.前言 二.产生原因 三.MultiDex的简要原理 四.MultiDex的使用 一.前言 首先说一下我遇到的情况,最近接手了一个项目是在已有的项目里进行更新添加一些功能,然后该项目导了N多 ...

最新文章

  1. Linux下网络socket编程——实现服务器(select)与多个客户端通信
  2. 【UVA - 227】Puzzle (模拟,水题)
  3. rocketmq 几种队列_关于RocketMQ的知识
  4. 新建jsp报错“The superclass javax.servlet.http.HttpServlet was not found on the Java Build Path”...
  5. ES6 class继承
  6. socket通信,server与多客户端通信(二)
  7. markdown公式编辑参考
  8. Mybatis-学习笔记(4)1对1、1对多、多对多
  9. VBS整人代码大合集
  10. 欢迎使用CSDN-markdown编辑器wewqewqew
  11. ssm mysql项目实战_ssm项目实战_ssm项目实战教程_ssm项目实战视频教程 _课课家
  12. java实现网页结构分析,网页列表发现
  13. WDF队列分析(3)
  14. 传智播客风清扬视频-------线程简介2
  15. c语言扑克牌同花顺比大小,为什么打扑克时“同花顺”最大
  16. uni-app中自定义图表(canvas实现chart图表)开发篇(5)-圆环进度条添加动画效果
  17. android系统关机广播,android关机方法汇总
  18. 基础平台项目之集成Jquery.pagination.js实现分页
  19. java计算机毕业设计飞机航班信息查询系统(附源码、数据库)
  20. 《战胜华尔街》书中的精髓:业余投资者如何根据行业特点选好股票,赚得比专业的投资者还要多?

热门文章

  1. C# 中 StringBuilder和String的区别
  2. 面试题01(C++)
  3. Xming + PuTTY 在Windows下远程Linux主机使用图形界面的程序
  4. [英]Promises Don't Come Easy
  5. java中自然排序和比较器排序
  6. 深度之眼课程打卡-python入门05
  7. 转载:独立思考能力吞噬
  8. ustc小道消息20220122
  9. [Python语音识别项目笔记] 3softmax函数
  10. 学长毕业日记 :本科毕业论文写成博士论文的神操作20170331