对于功能越来越复杂的app的两大问题

  • 一:我们自己应用的方法数超过了65536 
    我们所说的方法数限制,这个方法数包括了jar包,框架,还有我们自己应用的代码,当我们应用的代码超过65536时,结果如下:

    我们看到,显示我们方法的引用是65579.而引用数最大是65536,建议我们开启分包方案。

    二:我们应用的方法数没有超过65536,但是加上依赖的jar包,框架等,超过了65536(根据方案一的结果,我们应用方法数是65579,那我们删掉200个方法,就小于65536)

    报错如下:

  • 三:方法数并没有超过65536,编译也完成了,但是在android2.3以前的系统安装的时候,会异常中止安装。

    这个问题会发生在Android 2.2以及Android 2.3的设备上,涉及到一个名为dexopt的程序,全称dex optimization,即dex文件优化程序。在优化过程中,dexopt采用一个固定大小的缓冲区(LinearAlloc)来存储应用中所有方法的信息,那么之所以会出现在老版本停止安装,是因为老版本的缓冲区的大小是5M,而在新版本中,这个缓冲区的大小是8M或者16M,在老版本中更容易超过这个限制。

    dexopt的执行过程是在第一次加载dex文件的时候执行的。这个过程产生了一个ODEX文件,全称Optimised Dex。这个ODEX文件是在安装过程中从apk里提取出的可运行文件,是优化dex产生的,再把apk包里的dex文件删除,这样就做到了预先提取。如果没有ODEX文件,那么系统会从apk包中提取dex然后再运行。所以优化后可以加快软件的启动速度,预先提取,减少对RAM的占用。

    在早期的Android系统中,dexopt会把每一个类的方法id检索起来,存在一个链表结构里,而这个链表的长度是用一个short类型来保存的,导致方法id的数目不能够超过65536个。虽然新版本的android系统中,dexopt修复了这个问题,但是老版本的android系统的用户市场占有率还是占一定比例,还是不能放弃这部分用户的,所以我们在开发中需要对老版本的这个问题进行兼容。

2.方法数越界的解决方案

  • 插件化技术

    我们可以采用动态加载部分dex,通过将一个dex拆分成两个或多个dex,解决方法数越界的问题。

    插件化是一套重量级的技术方案,我们需要通过反射来调用插件的类或方法,要使用一套插件框架来配合,而且插件化适合一些独立的模块,兼容性问题往往较多,如果只是用于解决方法数越界的话,并不是最好的方案。

  • multidex解决方案

    为了解决方法数越界的问题,Google在2014年提出了multidex的解决方案,这个方案主要是针对AndroidStudio和Gradle编译环境的,将一个dex文件拆成两个或多个dex文件。

    不过需要注意的是multidex有一个版本问题,在Android 5.0以前使用multidex需要手动引入Google提供的android-support-multidex.jar这个jar包。这个jar包我们可以在Android SDK目录下的extras/android/support/multidex/library/libs下找到。而从Android 5.0开始,Andorid默认支持了multidex。

    所以,我们就需要注意我们的SDK版本了,如果已经支持了multidex,而我们又把android-support-multidex.jar放在了项目的libs文件下,就会报错。

3.在Gradle和代码中配置使用Multidex

  • 在Gradle中配置使用Multidex

    由于Android的Gradle插件在Android Build Tool 21.1开始支持使用multidex,所以我们需要使用Android Build Tools 21.1及以上版本,修改app目录下的build.gradle文件,有两点需要修改。

    (1)在defaultConfig中添加multiDexEnabled true这个配置项。 
    (2)在dependencies中添加multidex的依赖: 
    compile ‘com.android.support:multidex:1.0.0’

    注意buildToolsVersion要高于21.1,配置好如下: 

  • 在Gradle中配置好之后,我们还需要在代码中加入支持multidex的功能,有三种方案可选

    方案一:在manifest文件中指定Application为MultiDexApplication,如下:

    方案二:写一个Application类并继承MultiDexApplication,并在AndroidManifest.xml的application标签中进行注册(在application标签中增加name属性,并添加自己的Application类名即可),如果不是想重写MultiDexApplication中一些方法的话,还是方案一更方便些。如下:

    注册如下:

    方案三:如果不想按方案二继承,我们可以重写Application的attachBaseContext方法,注意,这个方法比onCreate方法先执行。具体方法是创建一个新类,继承Application,然后重写attachBaseContext方法,并在AndroidManifest.xml的application标签中进行注册(与方案二注册相同)如下:

    对于在AndroidManifest.xml中注册,与方案二的注册相同。

3.使用Multide分包后两种情况的结果

我们的Demo图如下,我们根据该图和dex文件反编译的结果分析分包情况。

  • 情况一:方法数没有越界 
    我们将方法数控制在65536以内,方法数没有越界的话,是不会分包的,解压apk,你会发现apk里只有一个classes.dex,如下

    将其反编译后(不知道怎么反编译的可以看一下这篇博文:http://blog.csdn.net/gaozhan_csdn/article/details/51984056),结果如下:

    可以发现,我们的类都在这个主dex文件里,并没有分包。

  • 情况二:方法数越界

    我们再将方法数增加到65536以上。解压apk,结果如下: 

    对三个dex文件反编译一下,看看它们里面分别都包含了什么类。 
    classes.dex(主dex)下的类视图: 

    1. classes2.dex的类视图

    1. classes3.dex的类视图

    可以发现Second类和Third类分别在classes2.dex文件和classes3.dex文件里,其他类都在主dex文件里(classes.dex),我们用multidex的确实现了分包从而解决了方法数越界的问题。

4.使用MultiDex存在的一些问题

1.Application 中的静态全局变量会比MutiDex的 instal()方法优先加载,所以建议避免在Application类中使用静态变量引用main classes.dex文件以外dex文件中的类。

或者这样解决:

一些在二级Dex加载之前,可能会被调用到的类(比如静态变量的类),需要放在主Dex中.否则会ClassNotFoundError. 通过修改Gradle,可以显式的把一些类放在Main Dex中.

  1. afterEvaluate {
  2. tasks.matching {
  3. it.name.startsWith('dex')
  4. }.each { dx ->
  5. if (dx.additionalParameters == null) {
  6. dx.additionalParameters = []
  7. }
  8. dx.additionalParameters += '--multi-dex'
  9. dx.additionalParameters += "--main-dex-list=$projectDir/<filename>".toString()
  10. }
  11. }

注意上面是修改后的Gradle,其中是一个文本文件的文件名,存放在和这个build.gradle脚本同一级的文件目录下,而不是 项目根目录。可以把这个文本文件起名为multidex.keep,内容如下.实际就是把需要放在Main Dex的类罗列出来.

  1. android/support/multidex/BuildConfig/class
  2. android/support/multidex/MultiDex$V14/class
  3. android/support/multidex/MultiDex$V19/class
  4. android/support/multidex/MultiDex$V4/class
  5. android/support/multidex/MultiDex/class
  6. android/support/multidex/MultiDexApplication/class
  7. android/support/multidex/MultiDexExtractor$1/class
  8. android/support/multidex/MultiDexExtractor/class
  9. android/support/multidex/ZipUtil$CentralDirectory/class
  10. android/support/multidex/ZipUtil/class

project.afterEvaluate标签在特定的project配置完成后运行,而gradle.projectsEvaluated在所有projects配置完成后运行。 注意afterEvaluate需要放在android{}里,不可放外面。

但是最新的as中,会自动判断依赖关系来分dex,比如以下application中:

  1. public class MyApp extends MultiDexApplication {
  2.  
  3. public static MutilTest5 mutilTest5 = new MutilTest5();
  4.  
  5. @Override
  6. public void onCreate() {
  7. super.onCreate();
  8. }
  9. }

默认情况下,本来MuitlText5要分到class2.dex里面去,但是因为app里静态变量需要用到MuitlText5,如果放到class2.dex中会找不到(因为app中静态变量初始化会在加载主dex文件之前执行),所以会自动放到主dex文件里去

但是如果依靠as自动分析,在你代码存在反射和native的情况下,也不保证100%正确,如果不正确,还是需要自己配置哪个类放到主dex中

Android分包方案multidex的更多相关文章

  1. android分包方案

    当一个app的功能越来越复杂,代码量越来越多,也许有一天便会突然遇到下列现象: 1. 生成的apk在2.3以前的机器无法安装,提示INSTALL_FAILED_DEXOPT 2. 方法数量过多,编译时 ...

  2. Android dex分包方案和热补丁原理

    一.分包的原因: 当一个app的功能越来越复杂,代码量越来越多,也许有一天便会突然遇到下列现象: 1. 生成的apk在2.3以前的机器无法安装,提示INSTALL_FAILED_DEXOPT 2. 方 ...

  3. [转]Android dex分包方案

    转载自:https://m.oschina.net/blog/308583 当一个app的功能越来越复杂,代码量越来越多,也许有一天便会突然遇到下列现象: 1. 生成的apk在2.3以前的机器无法安装 ...

  4. Android dex分包方案

    当一个app的功能越来越复杂,代码量越来越多,也许有一天便会突然遇到下列现象: 1. 生成的apk在2.3以前的机器无法安装,提示INSTALL_FAILED_DEXOPT 2. 方法数量过多,编译时 ...

  5. android 基于分包方案的修复

    # 本demo实现原理来自 https://github.com/dodola/HotFix https://zhuanlan.zhihu.com/p/20308548 # Anti类功能,及其原理 ...

  6. Android分包原理

    如果App引用的库太多,方法数超过65536后无法编译.这是因为单个dex里面不能有超过65536个方法.为什么有最大的限制呢,因为Android会把每一个类的方法id检索起来,存在一个链表结构里面. ...

  7. Android兼容包multidex的开发和构建方法

    在Android开发中,函数方法超过65k限制后,我们就常常会用到multidex分包解决,但是multidex的配置,对系统apk的构建.签名.打包复杂性大大的增加,严重的降低了构建效率.那这个问题 ...

  8. dex分包方案

    当一个app的功能越来越复杂,代码量越来越多,也许有一天便会突然遇到下列现象: 1. 生成的apk在2.3以前的机器无法安装,提示INSTALL_FAILED_DEXOPT 2. 方法数量过多,编译时 ...

  9. Android适应方案汇总(三)

    在Android适应方案汇总(一个).(两)在.我们理解一些基本概念. 那么详细的开发,我们应该重视起来. 首先,我们需要知道.关键的事实是,这两个适配器: (1).这点在单位的使用上用dp.sp以及 ...

随机推荐

  1. exit和return的区别

     主要有几下几个不同点:     1. return返回函数值,是关键字:exit是一个函数.     2. return是语言级别的,它表示了调用堆栈的返回:而exit是系统调用级别的,它表示了一个 ...

  2. orange pi pc 体验(一)

    最近在淘宝上看到一款和树莓派差不多的卡片机,定价才99元,而且是国产的,忍不住入手了一个,就是orange pi 感兴趣的可以百度搜索下,深圳一个公司出的,不过资料比树莓派少了很多,论坛中人也没多少, ...

  3. Windows下安装Python lxml库(无废话版)

    python官网:python-2.7.12.amd64.msihttps://pypi.python.org/pypi/setuptools:setuptools-28.6.0.zipsetupto ...

  4. js中解析json对象:JSON.parse()用于从一个字符串中解析出json对象, JSON.stringify()用于从一个对象解析出字符串。

    JSON.parse()用于从一个字符串中解析出json对象. var str = '{"name":"huangxiaojian","age&quo ...

  5. Instsrv.exe和Srvany.exe的使用方法

    要把应用程序添加为服务,你需要两个小软件:Instsrv.exe和Srvany.exe.Instsrv.exe可以给系统安装和删除服务,Srvany.exe可以让程序以服务的方式运行.这两个软件都包含 ...

  6. mysql 命令行操作1

    查看版本号 :SELECT @@version;或者status; 1.设置数据库系统的事务隔离级别 mysql> set global transaction isolation level ...

  7. ios cell时间相同隐藏算法

  8. Java c3p0连接池

    import java.beans.PropertyVetoException; import java.sql.Connection; import java.sql.SQLException; i ...

  9. AVD之PANIC: Could not open

    1 原因一:因为我们采用的是绝对路径定位,也就是说在环境变量里面把路径写死了,所以安装都不同,路径读不出来. 解决办法:①在环境变量中创建变量名:ANDROID_SDK_HOME,变量值:你当时安装S ...

  10. Linux -- objdump二进制文件比较

    objdump工具用来显示二进制文件的信息,就是以一种可阅读的格式让你更多地了解二进制文件可能带有的附加信息. 常用参数说明 -f 显示文件头信息 -D 反汇编所有section (-d反汇编特定se ...