java.lang.NoClassDefFoundError

作者: 唠嗑008 | 来源:发表于2017-08-03 21:06 被阅读125次

    这几天重构了公司项目,又引入了新的类库,结果杯具发生了。我的项目中加入了RXJava,RxAndroid,Retrofit来作为网络请求的。在运行中却出现了如下错误

    java.lang.NoClassDefFoundError: rx.subscriptions.CompositeSubscription
        at rx.internal.schedulers.CachedThreadScheduler$CachedWorkerPool.<init>(CachedThreadScheduler.java:60)
        at rx.internal.schedulers.CachedThreadScheduler.<clinit>(CachedThreadScheduler.java:42)
        at rx.plugins.RxJavaSchedulersHook.createIoScheduler(RxJavaSchedulersHook.java:90)
        at rx.plugins.RxJavaSchedulersHook.createIoScheduler(RxJavaSchedulersHook.java:76)
        at rx.schedulers.Schedulers.<init>(Schedulers.java:90)
        at rx.schedulers.Schedulers.getInstance(Schedulers.java:66)
        at rx.schedulers.Schedulers.io(Schedulers.java:160)
    

    这个错误,并且在几台测试机器上表现出的类名还不一样,在华为,魅族,OPPO上显示的都不一样,这就为难了,但是在我本人的三星机器(三星s6 6.0的系统)上就不会出现这种问题,这尼玛的,我就尴尬了,一时找不到原因。最后在StackOverflow找到了答案,发现是MultiDex出了问题。

    在重构中,添加了一些第三方类库,使得整个项目的方法数超出了65535,这个时候打包时本应该出现下面这个错误才对

    java.lang.IllegalArgumentException: method ID not in [0, 0xffff]: 65536
    at com.android.dx.merge.DexMerger$6.updateIndex(DexMerger.java:501)
    at com.android.dx.merge.DexMerger$IdMerger.mergeSorted(DexMerger.java:282)
    at com.android.dx.merge.DexMerger.mergeMethodIds(DexMerger.java:490)
    at com.android.dx.merge.DexMerger.mergeDexes(DexMerger.java:167)
    at com.android.dx.merge.DexMerger.merge(DexMerger.java:188)
    at com.android.dx.command.dexer.Main.mergeLibraryDexBuffers(Main.java:439)
    at com.android.dx.command.dexer.Main.runMonoDex(Main.java:287)
    at com.android.dx.command.dexer.Main.run(Main.java:230)
    at com.android.dx.command.dexer.Main.main(Main.java:199)
    at com.android.dx.command.Main.main(Main.java:103)
    

    但是这个项目并没有出现这个错误,导致我一开始根本没想到是这个分包的原因,因为这个项目的build.gradle文件沿用于一个项目的,在defaultConfig z中已经声明了 multiDexEnabled true ,当方法数超过65535时便会自动打出两个Dex包命名为 classes.dex classes2.dex ,一些方法被打入了第二个dex包,即classes2.dex中,导致了5.0以下机型无法运行应用报错,而6.0,7.0的手机则没有问题。

    解决方案

    • 1、在defaultConfig 中已经声明 multiDexEnabled true 用于启用MultiDex
    defaultConfig {
            applicationId "cn.jiayou.dsf.delivery"
            minSdkVersion 16
            targetSdkVersion 22
            versionCode 1
            versionName "1.0"
    
            multiDexEnabled = true //表示可以进行分包, 用于启用MultiDex
        }
    

    -2、.在依赖中添加如下依赖,支持包用于5.0以下系统

    compile 'com.android.support:multidex:1.0.1' 
    
    • 3、用MultiDexApplication替代Application,如果你的Application已经继承了其他类并且不想做改动,那么还有另外一种使用方式,覆写attachBaseContext()方法:
    public class MyApp extends Application{
        @Override
        protected void attachBaseContext(Context base) {
            super.attachBaseContext(base);
            MultiDex.install(this);
        }
    
         @Override
        public void onCreate() {
            super.onCreate();
          }
    }
    

    参考

    https://stackoverflow.com/questions/27698287/noclassdeffounderror-with-android-studio-on-android-4

    Android 使用android-support-multidex解决Dex超出方法数的限制问题,让你的应用不再爆棚

    相关文章

      网友评论

        本文标题:java.lang.NoClassDefFoundError

        本文链接:https://www.haomeiwen.com/subject/yvhmlxtx.html