64 K 限制

作者: GYLEE | 来源:发表于2017-06-05 17:01 被阅读52次

内容参考
内容来源
another

64 K 方法限制的原因

Android Project 经过编译打包,其中的 Java 代码(包括Library)转化为 DEX 格式字节码文件,这是 Android 5.0之前的 Dalvik 虚拟机决定的(5.0之后改为 ART 虚拟机),并且采用 short 类型引用 DEX 文件中的 method,这也为 method 数量的峰值大小埋下了隐患。short 类型能够表示的最大值是 65536,也就说单个 DEX 文件中最多只有 65536 个 method 能够得到引用,如果代码执行了超出部分的 method 引用,自然会报错,如 methodNotFound 等。1K 等于 1024,65536 刚好是 64K,为了便于称呼和使用,就将这个限制规则统称为 64K 方法数的引用限制。

解决 64 K 方法限制

为了解决 64 可方法数的限制,我们需要在项目中配置使用 Multidex ,当项目中的方法数超过 64 K 时,编译系统会自动编译出多个 Dex (Dalvik Executable)文件。

Android 5.0 之前版本的 Dalvik 可执行文件分包支持

Android 5.0(API 级别 21)之前的平台版本使用 Dalvik 运行时来执行应用代码。默认情况下,Dalvik 限制应用的每个 APK 只能使用单个 classes.dex 字节码文件。要想绕过这一限制,您可以使用 Dalvik 可执行文件分包支持库,它会成为您的应用主要 DEX 文件的一部分,然后管理对其他 DEX 文件及其所包含代码的访问。

注意:由于 Instant Run 机制利用的就是 multidex 原理,当项目中minSdkVersion 参数设置为20或者更小,并且运行在 Android 4.4 (API 20) 或更低版本的设备中时,Instant Run 将失效。

Android 5.0 及更高版本的 Dalvik 可执行文件分包支持

Android 5.0(API 级别 21)及更高版本使用名为 ART 虚拟机(Android RunTime),采用的是OAT技术(Ahead-of-time,预编译),后者原生支持从 APK 文件加载多个 DEX 文件。ART 在应用安装时执行预编译,扫描 classesN.dex 文件,并将它们编译成单个 .oat 文件,供 Android 设备执行。因此,如果您的 minSdkVersion 为 21 或更高值,则不需要 Dalvik Executable 文件分包支持库 。

注意:使用Instant Run时,如果项目中的 minSdkVersion 参数设为21或更高版本,Android Studio 编译运行时会自动使应用支持 multidex 。但 Instant Run 仅仅作用于 debug 版本,我们依然需要给 release 版本配置 multidex 来避开 64K 方法数的限制。

配置您的应用进行 Dalvik Executable 文件分包

将您的应用项目设置为使用 Dalvik 可执行文件分包配置需要对您的应用项目进行以下修改,具体取决于应用支持的最低 Android 版本。

如果您的 minSdkVersion 设置为 21 或更高值,您只需在模块级 build.gradle 文件中将 multiDexEnabled 设置为 true,如此处所示:

android {
    defaultConfig {
        ...
        minSdkVersion 21 
        targetSdkVersion 25
        multiDexEnabled true
    }
   
}

但是,如果您的 minSdkVersion设置为 20 或更低值,则您必须按如下方式使用 Dalvik 可执行文件分包支持库:修改模块级 build.gradle文件以启用 Dalvik 可执行文件分包,并将 Dalvik 可执行文件分包库添加为依赖项,如此处所示:

android {
    defaultConfig {
        ...
        minSdkVersion 15 
        targetSdkVersion 25
        multiDexEnabled true
    }
    ...
}
dependencies {
  compile 'com.android.support:multidex:1.0.1'
}

根据是否要替换 Application 类,执行以下操作之一:如果您没有替换 Application类,请编辑清单文件,按如下方式设置<application> 标记中的 android:name

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

如果您替换了 Application 类,请按如下方式对其进行更改以扩展 MultiDexApplication(如果可能):

public class MyApplication extends MultiDexApplication { ... }

或者,如果您替换了 Application 类,但无法更改基本类,则可以改为替换 attachBaseContext()方法并调用 MultiDex.install(this)来启用 Dalvik 可执行文件分包:

public class MyApplication extends SomeOtherApplication {
  @Override
  protected void attachBaseContext(Context base) {
     super.attachBaseContext(context);
     Multidex.install(this);
  }
}

有时候,你可能还需要改变一下 javaMaxHeapSize 的大小:

android {
    dexOptions {
        javaMaxHeapSize "4g"
    }
}

构建应用后,Android 构建工具会根据需要构建主 DEX 文件 (classes.dex) 和辅助 DEX 文件(classes2.de 和 classes3.dex 等)。然后,构建系统会将所有 DEX 文件打包到您的 APK 中。运行时,Dalvik 可执行文件分包 API 使用特殊的类加载器来搜索适用于您的方法的所有 DEX 文件(而不是仅在主 classes.dex文件中搜索)。

相关文章

  • 64K方法数限制原理与解决方案

    64K限制的原因 使用MultiDex解决64K限制的问题 如何避免出现64K限制未完待续

  • Android MultiDex

    什么是64K限制和LinearAlloc限制 64K限制 是什么原因导致方法数不能超过64K呢?(其实不仅仅是方法...

  • Android最大方法数和解决方案

    什么是64K限制和LinearAlloc限制 64K限制 随着Android应用功能的增加,代码量不断地增大,当应...

  • 64 K 限制

    内容参考内容来源another 64 K 方法限制的原因 Android Project 经过编译打包,其中的 J...

  • 关于Tinker热修复

    Android 应用的Dex有64K引用限制,超过限制后就不得不优化工程代码将方法数限制在64K以内或者使用分包方...

  • eclipse中multidex的使用

    [TOC] 64k方法数限制解决方案 Eclipse下Android 64k方法数的解决方案 eclipse An...

  • Android The number of method ref

    方法数64k的限制The number of method references in a .dex file c...

  • codeJan与青蛙——动态规划

    时间限制:C/C++ 1秒,其他语言2秒空间限制:C/C++ 262144K,其他语言524288K64bit I...

  • android MultiDex分包

    参考 64k限制 Dalvik Executable 规范将可在单个 DEX 文件内可引用的方法总数限制在 65,...

  • Android-65536方法数限制

    从编译apk 遇到65536(64k) 方法数限制问题说起。 一.方法数限制包含 1.android framew...

网友评论

    本文标题:64 K 限制

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