Android 分包导致报错ClassNotFoundException

      昨天中午发现自己现在的项目突然在三星手机上无法启动,一起动就无响应,平时用乐视手机(乐视是新机子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的完整路径

 

  • 5
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: Android应用程序中常见的类错误包括: 1. ClassNotFoundException:这种错误通常是由于应用程序中缺少依赖库或类文件而引起的。解决方法是检查项目的依赖关系,并确保所有必需的库和类文件都存在。 2. NoClassDefFoundError:这种错误通常是由于应用程序编译时未能正确包含类文件而引起的。解决方法是检查应用程序的编译设置并确保所有必需的类文件都被正确包含。 3. NoSuchMethodError:这种错误通常是由于应用程序使用了不存在的方法而引起的。解决方法是检查代码中的方法调用,并确保调用的方法存在于目标类中。 4. VerifyError:这种错误通常是由于应用程序包含了无效的字节码而引起的。解决方法是检查应用程序的编译设置并确保所有字节码都是有效的。 如果以上方法都不能解决问题,你可以尝试清除应用程序的缓存或重新安装应用程序。 ### 回答2: Android中的.class错误通常是由于编译错误或引入错误的问题引起的。在Android开发中,.class文件是Java编译器生成的字节码文件,它包含了编译后的Java代码。 首先,当我们在编译Android项目时,如果代码中存在语法错误、缺失的引入或其他编译错误,编译器会生成.class错误。解决这种错误的最佳方法是仔细检查代码,查找并修复所有的语法错误和缺失的引入。 另外,有时候我们可能会在项目中引入错误的库文件或依赖项,这也会导致.class错误。在这种情况下,我们需要检查项目的依赖项和库文件,并确保它们与我们的项目兼容和正确。 此外,还可能发生一些其他错误,如.class文件不存在、.class文件被误删等。在这种情况下,我们需要确认我们的项目中是否包含所需的.class文件,并在必要时进行恢复或重新构建。 总之,当遇到Android .class错误时,我们应该先检查代码的语法错误和引入错误,然后检查项目的依赖项和库文件是否正确,最后确认所需的.class文件是否存在。如果仍然无法解决问题,我们可以尝试重新构建项目或咨询开发社区寻求帮助。 ### 回答3: 当我们在编写Android应用程序时,如果我们在项目中使用了某个类文件,却出现了Android .class报错的情况,有一些可能的原因需要我们检查和解决。 首先,我们需要检查是否已正确导入所需的类文件。在Android开发中,我们通常使用import语句来导入所需的类,以便在代码中使用。如果我们忘记导入类文件或者导入错误的类文件,就会出现Android .class报错。我们可以通过检查import语句的正确性来解决这个问题。 其次,我们需要确保所使用的类文件已经存在于项目的类路径中。Android开发使用的类文件通常是在SDK中提供的库文件或我们自己创建的类文件。如果我们忘记将这些类文件包含在项目的类路径中,或者包含了错误的类文件,就会导致Android .class报错。我们可以通过检查项目的类路径配置来解决这个问题。 另外,Android .class报错还可能是因为我们正在使用的类文件版本与当前Android环境不兼容。在Android开发中,不同的Android版本可能会提供不同的类文件或类文件版本。如果我们使用了错误的Android类文件版本,就会出现Android .class报错。我们可以通过检查项目的Android目标版本和所使用的类文件版本来解决这个问题。 最后,我们还需要检查是否有重复的类文件或类文件命名冲突。在Android开发中,我们常常会使用不同的库文件或自定义类文件来实现功能。如果这些类文件之间存在重复或命名冲突,就会导致Android .class报错。我们可以通过检查项目的类文件命名和确保没有重复的类文件来解决这个问题。 总结来说,Android .class报错可能是由于导入错误、类文件不在类路径中、版本不兼容或命名冲突等原因所导致。我们可以通过检查导入语句、类路径配置、Android版本、类文件命名和重复文件来解决这个问题。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值