构建方法数超过65K的Android应用

44lufei44 9年前

来自: http://wangxinghe.me/blog/2016-02-04/android-method-over-65k/


1. 关于65535限制

通常一个APK文件中包含一个Dex文件,而一个Dex文件允许的最大方法数是65535。当APK很大,方法数很多时,可以将一个APK文件拆分为多个Dex的方式。 #### 2. 如何避免65535问题及MultiDex限制条件
(1)使用MultiDex方式

Android5.0之前的机制是Dalvik运行时(Dalvik Runtime),依赖库multidex support library作为主Dex的一部分用于管理从Dex

Android5.0及以后的机制是ART运行时(ART Runtime),ART运行时在程序安装阶段进行预编译,将所有Dex文件预编译为.oat文件

MultiDex使用方法如下:

在build.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'  }

然后配置Application

<?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>

或者extends MultiDexApplication,并在attachBaseContext() 方法中添加MultiDex.install(this)

(2)动态加载(后续研究)

限制条件:

multi support library在某些情况下可能存在问题,如:

  • 当从Dex过大时,在某些设备上安装时会引起ANR
  • 在Android4.0之前的设备上可能无法启动程序,必须减少方法数
  • 在Android5.0之前如果存在过大内存分配的case,根据Dalvik线性内存分配限制,会导致crash
  • 关于主Dex应包含哪些类会有复杂的需求

3. 开发小技巧

通常MultiDex都是很耗时的,因为构建系统会花较多时间来决定哪些类放在主Dex,哪些类放在从Dex。平常开发过程中,可以充分利用Android5.0的ART运行时方式进行开发,不过需要用5.0以上的设备。

ART运行时只进行如下操作:

  • 预分包阶段,将每个module构建成一个Dex文件
  • 将所有Dex文件按原样封装到APK
  • 不需要花时间重新决定哪些放到主Dex,哪些放到从Dex这一过程,因此节约很多时间

配置过程如下,平常构建dev即可,发布时使用prod:

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'  }    

IDE中可以先Sync一下工程,再在Build Variant选择特定的包进行构建:

pic

4. 测试

使用MultiDexTestRunner进行测试(在Gradle1.1之前,需要添加下面的dependencies),先进行配置:

android {    defaultConfig {        ...        testInstrumentationRunner "com.android.test.runner.MultiDexTestRunner"    }  }    dependencies {      androidTestCompile('com.android.support:multidex-instrumentation:1.0.1') {          exclude group: 'com.android.support', module: 'multidex'      }  }

再在MultiDexTestRunner的onCreate方法中添加如下代码:

public void onCreate(Bundle arguments) {      MultiDex.install(getTargetContext());      super.onCreate(arguments);      ...  }
本文原文发自 某学姐, 转载请保留出处, 谢谢.