ABI 管理
大家先来看看Google官方对Android ABI管理的介绍:不同 Android 手机使用不同的 CPU,因此支持不同的指令集。CPU 与指令集的每种组合都有其自己的应用二进制界面(或 ABI)。 ABI 可以非常精确地定义应用的机器代码在运行时如何与系统交互。 您必须为应用要使用的每个 CPU 架构指定 ABI。典型的 ABI 包含以下信息:机器代码应使用的 CPU 指令集。运行时内存存储和加载的字节顺序。可执行二进制文件(例如程序和共享库)的格式,以及它们支持的内容类型。用于解析内容与系统之间数据的各种约定。这些约定包括对齐限制,以及系统如何使用堆栈和在调用函数时注册。运行时可用于机器代码的函数符号列表 - 通常来自非常具体的库集。本页枚举了 NDK 支持的 ABI,并且提供每个 ABI 如何运行的信息。
支持的 ABI
每个 ABI 支持一个或多个指令集。表 1 提供每个 ABI 支持的指令集概览。
表 1. ABI 和支持的指令集。
ABI | 支持的指令集 | 说明 |
---|
armeabi | ARMV5TE 和更高版本Thumb-1 | 无硬浮点。 |
armeabi-v7a | armeabiThumb-2VFPv3-D16其他(可选) | 与 ARMv5、v6 设备不兼容。 |
arm64-v8a | AArch-64 |
x86 | x86 (IA-32)MMXSSE/2/3SSSE3 | 不支持 MOVBE 或 SSE4。 |
x86_64 | x86-64MMXSSE/2/3SSSE3SSE4.1、4.2POPCNT |
mips | MIPS32r1 及更高版本 | 使用硬浮点,并且假设 CPU:FPU 时钟比率为 2:1 以获取最大兼容性。 不提供 micromips 或 MIPS16。 |
mips64 | MIPS64r6 | |
Android 平台 ABI 支持
Android 系统在运行时知道它支持哪些 ABI,因为版本特定的系统属性会指示:设备的主要 ABI,与系统映像本身使用的机器代码对应。可选的辅助 ABI,与系统映像也支持的另一个 ABI 对应。此机制确保系统在安装时从软件包提取最佳机器代码。为实现最佳性能,应直接针对主要 ABI 进行编译。例如,基于 ARMv5TE 的典型设备只会定义主要 ABI:
armeabi
。 相反,基于 ARMv7 的典型设备将主要 ABI 定义为
armeabi-v7a
,而将辅助 ABI 定义为
armeabi
,因为它可以运行为每个 ABI 生成的应用原生二进制文件。许多基于 x86 的设备也可运行
armeabi-v7a
和
armeabi
NDK 二进制文件。对于这些设备,主要 ABI 将是
x86
,辅助 ABI 是
armeabi-v7a
。基于 MIPS 的典型设备只定义主要 ABI:
mips
。
安装时自动解压缩原生代码
安装应用时,软件包管理器服务将扫描 APK,查找以下形式的任何共享库:
lib/<primary-abi>/lib<name>.so
如果未找到,并且您已定义辅助 ABI,该服务将扫描以下形式的共享库:
lib/<secondary-abi>/lib<name>.so
找到所需的库时,软件包管理器会将它们复制到应用的
data
目录 (
data/data/<package_name>/lib/
) 下的
/lib/lib<name>.so
。如果根本没有共享对象文件,应用也会构建并安装,但在运行时会崩溃。
问题所在
Google官方已经说的很清楚了,安装apk时,会扫描对应abi版本的库,没有的话就扫描兼容的库,如果根本没有共享对象文件,应用也会构建并安装,但在运行时会崩溃。而我们在平时开发中经常会用到很多第三方库,但是他们提供的so文件支持版本,都会有很多不同,所以在构建的时候只能选择其中一种平台或者几种平台库打包。现在通常的做法就是只打包
armeabi-v7a一种平台的库,这种方式很好解决,直接使用Google官方提供的方案打包即可,方法如下:
1.在项目application model的build.gradle添加如下节点
android {
splits {
abi {
enable true
reset()
include 'armeabi-v7a'
universalApk true
}
}
}
参数解析
| 参数 | 释义 |
---|
splits | 拆分apk |
abi | 根据ABI拆分 |
enable | 是否启用拆分 |
include | 包含哪些ABI类型 |
universalApk | 是否生成通用的APK |
具体可以参照官网:https://developer.android.com/studio/build/configure-apk-splits.html#configure-abi-split这种方案有一个缺点:如果我想同事打包armeabi-v7a和x86的 so包到同一个apk,这种方式是无法做到的,因为它只能打包一种平台的so包到同一个apk中,你如果include后边跟了几个平台,他会单独打包成对应平台的apk。
我们要解决的问题就是:如何打包多个平台的so包到同一个apk中,我提供两种解决方案解决这个问题
第一种:
1.在项目application model的build.gradle添加如下节点
android {
defaultConfig {
ndk {
abiFilters 'armeabi' ,'armeabi-v7a'
}
}
}
然后在project的根目录(和local.properties同级)找到gradle.properties文本文件, 文件里面加入下面这行
android.useDeprecatedNdk=true
第二种:
1.在项目application model的build.gradle添加如下节点
android {
packagingOptions {
exclude 'lib/armeabi/**' //exclude 'lib/armeabi-v7a/**' exclude 'lib/arm64-v8a/**' exclude 'lib/mips/**' exclude 'lib/mips64/**' //exclude 'lib/x86/**' exclude 'lib/x86_64/**'
}
}
参数解析
| 参数 | 释义 |
---|
packagingOptions | 打包配置 |
exclude | 排除掉哪些文件 |
您想打包哪个平台的,就注释掉对应的行注:目前我只找到这两种方式,如果您有更好的方式,请留言。