昨天中午发现自己现在的项目突然在三星手机上无法启动,一起动就无响应,平时用乐视手机(乐视是新机子6.0,三星是11年的4.4.2),查看日志,折腾了一天时间,还好解决了,最终原因是分包出现问题,项目多次改版,也多半年了,方法数超64K(1K=1024,65536刚好是64K);
我多方面折腾,日志主要出现下图情况:
主要异常就是 java.lang.NoClassDefFoundError;java.lang.ClassNotFoundException:反正就是类找不到,明明清单里也写配置,项目也没报错,在乐视上还能正常运行,怎么就在三星上出问题了,在第二种异常on path: DexPathList[[zip 涉及到dex,估计和分包可能有关系吧,我把apk解压后,是有classes.dex和classes2.dex,分包成功呢,因为之前的方法数大于64K了,就采用分包,以为这里没问题,然而就坑在这了,分包是成功,但是Android系统在启动应用时加载dex,却并不同时加载这个两个classes.dex和classes2.dex,而是先加载主classes.dex,其他的.dex在应用启动后才进行动态加载安装,如果加载的类不在主dex,就会报错找不到类ClassNotFoundException错误;
解决方案
1、在module下的build.gradle文件 ,添加multiDexEnabled true
defaultConfig {
applicationId "com.kakaxi.xx"
minSdkVersion 14
targetSdkVersion 22
versionCode 1
versionName "1.0.1"
multiDexEnabled true
testInstrumentationRunner "android.test.InstrumentationTestRunner"
2、在android节点内部添加jumboMode = true
dexOptions {
javaMaxHeapSize "4g"
jumboMode = true
}
3、添加分包依赖,谷歌官方推出了multidex兼容包,配合AndroidStudio实现了一个APK包含多个dex的功能
compile 'com.android.support:multidex:1.0.1'
4、让自己的Application继承MultiDexApplication类,或者在Application下重新attachBaseContext(Context base)方法,初始化 MultiDex.install(this);如何下图
@Override
protected void attachBaseContext(Context base) {
super.attachBaseContext(base);
MultiDex.install(base);
}
选择一种方式即可。
这样当方法数超过64K时,进行分包,出现类找不到的问题就解决了;
而自己犯的错就是在Application没有处理,也想不起半年前,自己参考哪篇文章,只在gradle里做了配置,application里没做处理,或许当时出现的问题,那个做就OK了,东西还是要理解全面滴,要不不断填坑,不同系统版本的手机在处理多个dex是不一样.项目随意业务增加,改变,加上引用的库增加,方法数很容易超过64k,而单个dex能接纳的方法数不能超过65536个,有这个限制是因为Android会把每个类里的方法统计起来,存在一个short类型为长度的单链表里,short占用两个字节,-2的15次方到2的15次方,即-32768到32768,总保持数量为65536,Android在新的系统5.0里修复了这个问题,但是为了兼容低版本,扔需要处理;
Android 5.0之前,安卓系统采用的是Dalvik虚拟机,采用的是JIT技术(Just-in-time compilation,即时编译,运行时编译DEX字节码文件,这也是以前为什么安卓手机用户总是诟病Android系统比iOS系统运行卡顿的原因),限制每个APK文件只能包含一个DEX文件(即classes.dex)。为了绕开这个限制,Google给我们提供了multidex support library兼容包,帮助我们实现应用程序加载多个DEX文件,并且这个兼容包作为程序的主DEX文件,管理者其他DEX文件的访问。
Android 5.0之后,安卓系统改用了ART虚拟机(Android RunTime),采用的是OAT技术(Ahead-of-time,预编译,在应用安装的时候扫描应用中的所有DEX文件,并编译成一个.oat格式的文件供安卓设备执行,所以相比Dalvik虚拟机下的应用,安装时间较长)。因此可以理解为,使用ART虚拟机下的安卓系统自动支持APK文件中多个DEX的加载。所以我用乐视手机(Android 6.0)上apk正常运行,而在三星(Android4.4.2)却无法运行,报找不到类;
此外,有没有办法指定某些类被分包到主dex呢?有,在app目录下创建一个maindexlist.txt,我们在这个txt里将我们想要放在主dex中的类写进去即可,为了方便减少错误的出现,可以在在\app\build\intermediates\multi-dex\debug目录下找到了一个maindexlist.txt,注意,这个你直接在改了没用,一运行又恢复了,你要的做的是将这个maindexlist.txt复制到app目录下,在进行添加添加指定类即可,
添加指定类后,可以通过对apk进行解压,对主dex进行反编译进行查看,会发现刚才添加的指定类在里面;
如何查看apk方法数
方法一:dexdump -f apkName.apk | findstr method_ids_size
dexdump.exe直接用android_sdk\build-tools下的,用高版本下的,低版本有的有问题,dexdump.exe和apk放同一目录下,方式运行;
方法二:采用dex-method-counts.jar
java -jar dex-method-counts.jar apkName.apk
注意dex-method-counts.jar 和apk的完整路径