这几天重构了公司项目,又引入了新的类库,结果杯具发生了。我的项目中加入了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