亦余心之所善兮,虽九死其犹未悔。这篇文章主要讲述AndroidStudio利用android-support-multidex解决65536问题64k问题相关的知识,希望能为你提供帮助。
androidStudio利用android-support-multidex解决65536问题64k问题
版权声明:转载请注明本文转自严振杰的博客: http://blog.csdn.net/yanzhenjie100364k的各种异常
Error:The number of method references in a .dex file cannot exceed 64K.
Learn how to resolve this issue at https://developer.android.com/tools/building/multidex.html
或者
finished with non-zero exit value 2
当你的应用程序和库引用达到一定规模,你遇到构建错误显示你的应用已经达到了一个Android应用程序构建体系结构的限制。早期版本的构建系统报告这个错误如下:
Conversion to Dalvik format failed:
Unable to execute dex: method ID not in [0, 0xffff]: 65536
- 1
- 2
UNEXPECTED TOP-LEVEL EXCEPTION:
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)
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
trouble writing output:
Too many field references: 131000;
max is 65536.
You may try using --multi-dex option.
- 1
- 2
- 3
Error:The number of method references in a .dex file cannot exceed 64K.
- 1
关于64 k引用限制Android应用程序(APK)在Dalvik可执行文件的形式包含可执行的字节码文件(DEX)文件,其中包含已编译的代码来运行你的应用程序。Dalvik可执行规格限制一个Dex文件包含65536个方法:包括Android框架方法、Library方法的总数、和你自己的代码方法总数。因为65536等于64×1024,这一限制被称为“64k引用限制”。
这个极限就要求我们配置应用程序的构建过程,需要生成多个DEX文件,所以称为multidex 配置。
分析原因与注意事项解决方法分Android 5.0及以上系统和5.0以下系统怎么做。客官们不要着急,先看我一个个分析原因,毕竟我要装下逼哈哈。
一、Android 5.0以下的版本Android 5.0(API leve 21)之前的系统使用Dalvik执行应用程序代码。默认情况下,Dalvik限制一个apk只有一个Dex文件。为了绕过这个限制, 我们可以使用multidex support library,它成为我们APK的主要DEX文件的一部分,负责管理我们APK访问其他DEX文件和代码。
注意: 如果咱的项目minSdkVersion是20或更低,运行到Android 4.4(API leve 20)或者更低版本的设备上时需要禁用AndroidStudio的即时运行
二、Android 5.0和更高版本Android 5.0(API leve 21)和更高的系统使用runtime是ART ,原生支持从应用的apk文件加载多个DEX文件。ART在安装应用时预编译应用程序,会扫描多个classes(..N).dex文件编译成一个.oat的文件。更多Android5.0 runtime的更多信息,请参见即时运行-instant-run。
注意: 如果你使用即时运行 , AndroidStudio自动配置你的应用程序,你应用程序的minSdkVersion应该设置为21或更高。因为即时只工作在你APP的Debug版本,你任然需要配置你的release版本构建时用multidex避免64k的限制。
尽量避免64k限制在配置我们的App启用64k或者更多方法引用之前,我们可以减少应用代码内的调度总数,包括我们自身应用的方法和第三方的库,下面的几个策略或许可以帮到你:
使用上面的方法可以帮助我们避免在应用程序中生成太多无用的方法和减小我们apk的大小,这对于用自己服务器做app更新升级的同学是非常有帮助的。
- 检查你的APP的直接和间接的过度依赖关系:有时候我们用到某个Libaray的某几个方法或者功能时这个库非常大,减少这种依赖可能对与避免64k的问题非常有效。
- 在正式打包构建的时候,使用代码混淆器ProGuard混淆移除未使用的代码,也就是不把没有使用的代码打包到我们的apk中。
这里给大家推荐下任玉刚同学的插件式开发框架:https://github.com/singwhatiwanna/dynamic-load-apk,共同参与开发者:田啸,宋思宇。
解决64k问题在
Android SDK Build Tools 21.1
或者更高版本的build工具中用Android plugin gradle
。确保你更新Android SDK build tools
和Android support
到最新版本,然后用multidex配置应用程序。我们必须要做两步。第一步,修改主module的build.gradle文件在gradle中依赖
multidex
,并启用multiDexEnable
:android {
compileSdkVersion 21
buildToolsVersion defaultConfig {
...
minSdkVersion 14
targetSdkVersion 21
...// Enabling multidex support.
multiDexEnabled true
}
...
}dependencies {
compile ‘com.android.support:multidex:1.0.1‘
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
android.support.multidex.MultiDexApplication
类
两种情况第一种情况,如果我们的APP没有重写过Application类,我们直接继承
MultiDexApplication
,然后在manifest.xml中注册Application即可。第二种情况,如果我们已经重写过Application类,重写
attachBaseContext(Context)
方法,并调用MultiDex.install(this);
即可:protected void attachBaseContext(Context base) {
super.attachBaseContext(base);
MultiDex.install(this);
- 1
- 2
- 3
MultiDexApplication
的源码了,里面就是重写了这个方法而已哈哈:public class MultiDexApplication extends Application {
public MultiDexApplication() {
}protected void attachBaseContext(Context base) {
super.attachBaseContext(base);
MultiDex.install(this);
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
<
?xml version= encoding=?>
<
manifest xmlns:android=
package=>
<
application
...
android:name="刚才重写的Application全类名">
...
<
/application>
<
/manifest>
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- DEX文件安装到设备的过程非常复杂,如果第二个DEX文件太大,可能导致应用无响应。此时应该使用ProGuard减小DEX文件的大小。
- 由于Dalvik linearAlloc的Bug,应用可能无法在Android 4.0之前的版本启动,如果你的应用要支持这些版本就要多执行测试。
- 同样因为Dalvik linearAlloc的限制,如果请求大量内存可能导致崩溃。Dalvik linearAlloc是一个固定大小的缓冲区。在应用的安装过程中,系统会运行一个名为dexopt的程序为该应用在当前机型中运行做准备。dexopt使用LinearAlloc来存储应用的方法信息。Android 2.2和2.3的缓冲区只有5MB,Android 4.x提高到了8MB或16MB。当方法数量过多导致超出缓冲区大小时,会造成dexopt崩溃。
- Multidex构建工具还不支持指定哪些类必须包含在首个DEX文件中,因此可能会导致某些类库(例如某个类库需要从原生代码访问Java代码)无法使用。
UNEXPECTED TOP-LEVEL EXCEPTION:
com.android.dex.DexException: Library dex files are not supported in multi-dex mode
- 1
- 2
android {
...
dexOptions {
preDexLibraries = false
}
...
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
UNEXPECTED TOP-LEVEL ERROR:
java.lang.OutOfMemoryError: Java heap space
- 1
- 2
maxProcessCount 4 // this is the default value
javaMaxHeapSize "2g"
- 1
- 2
在Android leve 21或者更高SDK版本。使用ART-supported格式生成multidex输出更快,为我们节省时间,所以我们不必在调试的使用也兼容到5.0以下,所以我们配置最低版本的时候做个如下兼容:
android {
productFlavors {
// 自定义偏好设置.
dev {
// 在Android leve 21或更高版本编译更快
minSdkVersion 21
}
prod {
// 真正的生产环境.
minSdkVersion 14
}
}
...
}
dependencies {
compile ‘com.android.support:multidex:1.0.1‘
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
dependencies {
androidTestCompile(‘com.android.support:multidex-instrumentation:1.0.1‘) {
exclude group: ‘com.android.support‘, module: ‘multidex‘
}
}
- 1
- 2
- 3
- 4
- 5
官方参考文档:https://developer.android.com/tools/building/multidex.html
推荐阅读
- Android网络编程之使用HttpClient和MultipartEntityBuilder 批量同时上传文件和文字
- java.lang.UnsupportedClassVersionError: com/android/build/gradle/AppPlugin : Unsupported major.minor
- AndroidManifest.xml中的注册组件
- React Native DEMO for Android
- Android之聊天室设计与开发
- Android java.lang.IllegalStateException: Already logged in to server.
- Android 安全 App发布之后收集bug信息--腾讯Bugly的使用
- AndroidStudio怎么将开源项目发布到jcenter
- Android Studio常用插件