http://blog.csdn.net/daitu_liang/article/details/72987378
昨天中午发现自己现在的项目突然在三星手机上无法启动,一起动就无响应,平时用乐视手机(乐视是新机子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进行反编译进行查看,会发现刚才添加的指定类在里面;
分享到:
相关推荐
"java.lang.ClassNotFoundException: com.microsoft.jdbc.sqlserver.SQLServerDriver" 解决方案 [Microsoft][SQLServer 2000 Driver for JDBC]Error establishing socket. 解决了jsp连接 sql server 2000的问题
Android Caused by: java.lang.ClassNotFoundException解决办法 ...我想大家经常因为疏忽出现ClassNotFoundException的异常,这个异常在Android开发中无非就是告诉你类没有找到,那么什么原因导致没找到呢? 其
这个是我调用百度定位SDK做的一个小案例,但是在我电脑上能够运行,在其他人的电脑上就会报错,出现ClassNotFoundException异常。
ClassNotFoundException终极解决办法 完美解决ClassNotFoundException,jar包在线查找库,在线下载库
接口调用报错:java.net.SocketException: java.lang.ClassNotFoundException: Cannot find the specified class com.ibm.websphere.ssl.protocol.SSLSocketFactory
JSON转化时出现 java.lang.ClassNotFoundException: net.sf.ezmorph.MorpherRegistry
一:查看你下载的MySQL和mysql-connector-java,如果不对应的话肯定会报错。 对应规则: mysql-connector-java-5 Connector/J version Driver Type JDBC version MySQL Server version Status 5.1 4 3.0, 4.0...
java 9 中引入了模块的概念,默认情况下,Java SE中将不再包含java EE 的Jar包 而在 java 6/7 / 8 时关于这个API 都是捆绑在
JSP中ClassNotFoundException异常的解决办法.pdf 学习资料 复习资料 教学资源
java 找不到指定类时发生的异常(ClassNotFoundException) java 找不到指定类时发生的异常(ClassNotFoundException)
NULL 博文链接:https://sw1982.iteye.com/blog/607325
2. 一般启动报错: ClassNotFoundException异常 少jar包 3. 访问404,什么原因 4. 点击某个功能报错
本地运行报NoSuchMethodError,ClassNotFoundException。明明在依赖里有这个Jar包啊。怎么运行不了!? 项目中明明定义着某个jar包版本为2.0.2,怎么打包之后变成2.5.0了!? A项目引xxx.jar包运行好好的,B项目同样...
* 反序化的包名,类名,版本ID必须一致,否则ClassNotFoundException */ InputStream stream = this.context.getAssets().open("china-city.obj"); china = Country.loadFromStream(stream); //...
整个系统都没有错误,但是在发布运行后,控制台上却抛出了这样的异常: java.lang.ClassNotFoundException: org.apache.commons.dbcp.BasicDataSource
NULL 博文链接:https://redarmychen.iteye.com/blog/463864
NULL 博文链接:https://chimpp55.iteye.com/blog/2344330
struts初学者的数据池配置出现java.lang.ClassNotFoundException: org.apache.commons.dbcp.BasicDataSource异常的解决办法
MQTTv5版的客户端库,原地址:... 本资源合入了解决低于android7.0报错Caused by: java.lang.ClassNotFoundException: Didn't find class "javax.net.ssl.SNIHostName"的问题。