Android应用使用Multidex突破64K方法数限制

写在前面

前几天,开发中遇到一个问题,Log信息如下:

E/AndroidRuntime(10943): FATAL EXCEPTION: main
E/AndroidRuntime(10943): Process: com.freeme.gallery, PID: 10943
E/AndroidRuntime(10943): java.lang.NoClassDefFoundError: com.freeme.gallery.data.DataManager$DateTakenComparator
E/AndroidRuntime(10943):     at com.freeme.gallery.data.DataManager.<clinit>(DataManager.java:65)
E/AndroidRuntime(10943):     at com.freeme.gallery.app.GalleryAppImpl.getDataManager(GalleryAppImpl.java:77)
E/AndroidRuntime(10943):     at com.freeme.gallery.provider.GalleryProvider.onCreate(GalleryProvider.java:101)
E/AndroidRuntime(10943):     at android.content.ContentProvider.attachInfo(ContentProvider.java:1656)
E/AndroidRuntime(10943):     at android.content.ContentProvider.attachInfo(ContentProvider.java:1627)
E/AndroidRuntime(10943):     at android.app.ActivityThread.installProvider(ActivityThread.java:5060)
E/AndroidRuntime(10943):     at android.app.ActivityThread.installContentProviders(ActivityThread.java:4634)
E/AndroidRuntime(10943):     at android.app.ActivityThread.handleBindApplication(ActivityThread.java:4567)
E/AndroidRuntime(10943):     at android.app.ActivityThread.access$1500(ActivityThread.java:153)
E/AndroidRuntime(10943):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1404)
E/AndroidRuntime(10943):     at android.os.Handler.dispatchMessage(Handler.java:110)
E/AndroidRuntime(10943):     at android.os.Looper.loop(Looper.java:193)
E/AndroidRuntime(10943):     at android.app.ActivityThread.main(ActivityThread.java:5351)
E/AndroidRuntime(10943):     at java.lang.reflect.Method.invokeNative(Native Method)
E/AndroidRuntime(10943):     at java.lang.reflect.Method.invoke(Method.java:515)
E/AndroidRuntime(10943):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:835)
E/AndroidRuntime(10943):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:651)
E/AndroidRuntime(10943):     at dalvik.system.NativeStart.main(Native Method)

从报错信息来看,是没有找到DateTakenComparator这个内部类且又是运行时异常,那是不是和ClassLoader有关系呢?
那么首先排除代码原因,开始从Gradle和Gradle插件版本入手,通过改变版本来验证。然而验证下来发现与Gradle并没关系。

那么问题到底出在哪呢?
没辙!于是开始按节点排查,排查过几个关键节点后,终于得出一个结论:引入某个特定library后就会报这个错

然而这个library是直接从Maven导入的,library本身肯定没有问题。似乎到这里线索又断了...恰逢此时,同事建议看下apk包大小。不看不知道,看过才恍然大悟,apk内大有乾坤啊。

apk包中含有两个.dex文件:classes.dexclasses2.dex,再看java.lang.NoClassDefFoundError,结果显而易见,方法数超限了!但是已经在build.gradle中配置了multiDexEnabled true和添加了android.support.multidex,为何还会出错呢? 原来是忘了继承MultiDexApplication了!敲脑袋ing...

接下来,我们借助官方文档来了解下64K方法数限制。

正文

随着应用不断增加新功能,引入新库,apk会越来越大,到达一定规模后就可能遇到方法数超限问题。
早期版本错误信息如下:

Conversion to Dalvik format failed:
Unable to execute dex: method ID not in [0, 0xffff]: 65536

较新版本错误信息如下:

trouble writing output:
Too many field references: 131000; max is 65536.
You may try using --multi-dex option.

其中数字65536是关键,Android平台的Java虚拟机Dalvik执行Dex程序时,使用的是short类型来索引DEX文件中的方法。这就意味着单个Dex文件可被引用的方法总数被限制为64x1024, 即65536。其中包括:

  • Android Framework的方法
  • library的方法
  • 我们自己写的方法

为突破这个限制,需要使用multidex来生成多个dex文件。

Android5.0 (API level 21)之前版本支持Multidex

Android5.0之前使用Dalvik运行时执行应用代码,默认Dalvik限制每个apk只能有一个字节码classed.dex文件。为突破这个限制,可以使用multidex support library来管理额外的dex文件(包括代码)。

Android5.0及更高版本支持Multidex

Android5.0及更高版本使用支持从apk中加载多个dex文件的ART运行时机制,在应用安装时,加载classed(...N).dex文件并编译成一个.oat文件以支持在Android设备上运行。关于Android 5.0运行时详见ART介绍

Note: While using Instant Run, Android Studio automatically configures your app for multidex when your app's minSdkVersion is set to 21 or higher. Because Instant Run only works with the debug version of your app, you still need to configure your release build for multidex to avoid the 64K limit.

如果使用Instant Run,当app的minSdkVersion大于或等于21时,Android Studio会自动配置支持multidex,但是仅debug版本有效,release版仍然需要配置multidex来突破64K限制。

避免64K限制

在配置multidex之前,你或许可以通过以下方法来减小方法总数(包括引用的、library里的和自己写的方法)。

  • 排除未使用的依赖 -此步骤通常能有效避免64K限制。
  • 使用ProGuard去除未使用的方法 -为release版本配置ProGuard,能有效排除一些无用方法

使用以上技术能有效避免更改构建配置来引用更多的方法,同时能减小apk大小,使用户消耗更少的流量。

使用Gradle配置Multidex

Android SDK Build Tools 21.1或更高版本上支持multidex,确定要配置multidex前请确保Android SDK Build ToolsAndroid Support Repository更新到较新版本。

通过以下步骤配置multidex:

  • 更改Gradle配置来支持multidex
  • 修改manifest。使其支持multidexapplication类

修改模块级builde.gradle文件,修改如下:

android {
    compileSdkVersion 21
    buildToolsVersion "21.1.0"

    defaultConfig {
        ...
        minSdkVersion 14
        targetSdkVersion 21
        ...

        // Enabling multidex support.
        multiDexEnabled true
    }
    ...
}

dependencies {
  compile 'com.android.support:multidex:1.0.0'
}

在manifest文件中,添加MultidexApplication Class的引用,如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.android.multidex.myapplication">
    <application
        ...
        android:name="android.support.multidex.MultiDexApplication">
        ...
    </application>
</manifest>

通过以上步骤即可支持multidex。

Note: If your app uses extends the Application class, you can override the attachBaseContext() method and call MultiDex.install(this) to enable multidex. For more information, see the MultiDexApplication reference documentation.

如果你的应用中已经继承Application,那么可以通过复写attachBaseContext()方法并调用MultiDex.install(this)来支持multidex,即无需修改manifest文件。更多信息请看MultiDexApplication

补充:
亦可直接将继承Application 改为继承MultiDexApplication,而无需修改manifest文件或复写attachBaseContext()方法。

multidex support library的使用限制

multidex support library有一些已知的限制请务必知晓,需要在应用时先行测试。

  • 如果classes2.dex文件较大,安装dex文件到设备的数据区是一个复杂的过程,可能会导致应用程序无响应(ANR)的错误。在这种情况下,应该使用ProGuard尽量减小dex文件的大小且删除无用的代码。

  • 在Android 4.0(API Level 14)之前,由于Dalvik linearalloc bug(问题22586),multidex可能是运行失败。如果希望运行在Level 14之前的Android系统版本,请先确保完整的测试和使用。优化代码可以减少或可能消除这些潜在的问题。

  • 应用程序使用了multiedex配置,会造成申请很大的内存分配。可能还会引起Dalvik虚拟机的崩溃(问题78035)。此分配限制是在Android 4.0 (API level 14)上增加的,但Android5.0 (API level 21)之前的版本仍有此限制。

  • multidex构建工具不支持指定哪些类必须包含在首个dex文件中,因而可能导致某些library无法使用。

优化Multidex的开发和构建

multidex会加长构建应用的时间,这个必要的过程可能会拖慢你的开发进度。
为加速构建过程,我们可以在Gradle中配置productFlavors: a development flavor and a production flavor.

开发时将minSdkVersion改为21使用ART运行时机制,这样能加快构建速度。release时改为合适的minSdkVersion,这样仅在release时费时较长。

build.gradle配置如下:

android {
    productFlavors {
        // Define separate dev and prod product flavors.
        dev {
            // dev utilizes minSDKVersion = 21 to allow the Android gradle plugin
            // to pre-dex each module and produce an APK that can be tested on
            // Android Lollipop without time consuming dex merging processes.
            minSdkVersion 21
        }
        prod {
            // The actual minSdkVersion for the application.
            minSdkVersion 14
        }
    }
          ...
    buildTypes {
        release {
            runProguard true
            proguardFiles getDefaultProguardFile('proguard-android.txt'),
                                                 'proguard-rules.pro'
        }
    }
}
dependencies {
  compile 'com.android.support:multidex:1.0.0'
}

完成上述配置后,你可以使用结合了dev productFlavorbuildType属性的devDebug变体app。
这个变体app包含如下特性:

  • 关闭了混淆(proguard)
  • 支持multidex
  • minSdkVersion 设置为 Android API level 21.

这些设置将使Gradle插件做如下事情:

  1. 编译应用的每个模块(包括依赖)为独立的dex文件,这个过程称为pre-dexing
  2. 不作修改地include每个dex文件到apk里
  3. 更重要的是,这些模块dex文件将不会合并,这样避免分割主dex文件,以加快速度

值得注意的是:上述配置后的devDebug变种app仅能运行在Android 5.0设备上

同时,你也可以构建其他变体app,也可以在终端使用gradel命令来实现多渠道打包等。更多有关flavorsGradle tasks信息, 请看Gradle Plugin User Guide(中文翻译).

在Android Studio中构建变种App

使用multidex时,构建变体app对管理构建过程是非常有用的。Android studio允许用户自己选择。

在Android Studio中构建变体app,步骤如下:

  1. 从左边栏打开Build Variants窗口
  2. 点击build variant以选择不同变体,如图:

测试Multidex应用

测试multidex应用,需在build.gradle中配置MultiDexTestRunner:

android {
  defaultConfig {
      ...
      testInstrumentationRunner "com.android.test.runner.MultiDexTestRunner"
  }
}

Note: With Android Plugin for Gradle versions lower than 1.1, you need to add the following dependency for multidex-instrumentation:

若Gradle插件版本低于1.1,你还需添加multidex-instrumentation依赖:

dependencies {
    androidTestCompile('com.android.support:multidex-instrumentation:1.0.1') {
         exclude group: 'com.android.support', module: 'multidex'
    }
}

备注:文中链接为官方链接,请爬墙观看!

参考资料
1.Building Apps with Over 64K Methods

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,963评论 25 707
  • 随着安卓平台的不断发展与壮大,市场上大而全的应用比比皆是,产品需求的变更累积和UI交互的极致追求,除了 resou...
    亦枫阅读 3,254评论 5 45
  • 这一章主要针对项目中可以用到的一些实用功能来介绍Android Gradle,比如如何隐藏我们的证书文件,降低风险...
    acc8226阅读 7,580评论 3 25
  • 随着安卓平台的不断发展与壮大,市场上大而全的应用比比皆是,产品需求的变更累积和UI交互的极致追求,除了 resou...
    阿俊贰阅读 870评论 0 14
  • 梧桐叶又落 在还未储藏够过冬的粮草 八月的丧钟就已敲响 生命如此黯淡 一场山的倾倒就可碾压 一阵风的放肆就会坠落 ...
    footprint123阅读 230评论 0 1