Android 分包导致报错ClassNotFoundException
来源:互联网 发布:淘宝企业店铺怎么操作 编辑:程序博客网 时间:2024/06/06 14:02
昨天中午发现自己现在的项目突然在三星手机上无法启动,一起动就无响应,平时用乐视手机(乐视是新机子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);如何下图@Overrideprotected 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进行反编译进行查看,会发现刚才添加的指定类在里面;
- Android 分包导致报错ClassNotFoundException
- Android 分包导致低于5.0版本报错ClassNotFoundException
- Android Trick 14: proguard导致签名后的apk报错:ClassNotFoundException 的问题解决
- android 老是报ClassNotFoundException
- 使用RecyclerView报错:ClassNotFoundException android.support.v4.util.ArrayMap
- android build.prop导致GooglePlay报错
- Tomcat启动报错,ClassNotFoundException
- Tomcat启动报错,ClassNotFoundException
- Oozie报错ClassNotFoundException:…
- 解决Android报java.lang.ClassNotFoundException问题
- android分包
- android分包
- android分包
- 当dex分包遇上NoClassDefFoundError&ClassNotFoundException
- 当dex分包遇上NoClassDefFoundError&ClassNotFoundException
- 当dex分包遇上NoClassDefFoundError&ClassNotFoundException
- Eclipse报错:java.lang.ClassNotFoundException: ContextLoaderListener
- Eclipse报错:java.lang.ClassNotFoundException: ContextLoaderListener
- Android自定义网络请求状态组件
- centos 6.5安装erlang 18.3
- 利用反射拼接sql语句
- 求三角形内任意一点到另一个三角形上的映射坐标点
- 在Ubuntu中安装Flawfinder
- Android 分包导致报错ClassNotFoundException
- Javascript触发事件列表
- crypt1-section1.3
- Java的日期与MySQL数据库日期的转化问题
- Android6.0/7.0新特性
- java中用Dom4J来操作xml文件(删除和修改和添加)
- 剑指offer——数据流中的中位数___ (堆应用实例)
- Log4j2分析与实践
- 双飞翼布局