應用程序二進制接口(Application Binary Interface)定義了二進制文件(尤其是.so文件)如何運行在相應的系統平台上,從使用的指令集,內存對齊到可用的系統函數庫。在Android 系統上,每一個CPU架構對應一個ABI:armeabi,armeabi-v7a,x86,mips,arm64- v8a,mips64,x86_64。
so文件和jar包存放到工程的libs目錄下。p.s.另外,可以將apk解壓出來,查看lib目錄,也能分析該apk的so文件目錄。
使用 app源根文件夾下build.gradle文件的配置:
android { compileSdkVersion 23 buildToolsVersion "23.0.2" defaultConfig { applicationId "com.lance.ndkexample" minSdkVersion 14 targetSdkVersion 23 versionCode 1 versionName "1.0" // 不聲明ndk標簽,項目默認會創建一個libapp.so的文件 ndk { // 聲明創建so庫的文件名,會自動添加lib前綴, 添加了前綴,不會自動添加 moduleName "MathKit" //聲明啟用Android日志, 在c/c++的源文件中使用的#include <android/log.h> 日志將得到輸出 ldLibs "log" // 聲明創建指定cpu架構的so庫, 不聲明的話, 默認(gradle 1.5.0)會生成4中架構 多一種mips架構 // 具體cpu架構的區別請參考: // for detailed abiFilter descriptions, refer to "Supported ABIs" @ // https://developer.android.com/ndk/guides/abis.html#sa abiFilters "armeabi", "armeabi-v7a", "x86" } } buildTypes { release { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } sourceSets { main { // 1. 配置在根目錄libs下可以加載第三方so庫, (最好不要創建jniLibs, 在眾多的開源庫中可能會引起沖突,還沒發現) // 2. 運行時會自動將libs目錄下的so庫拷貝到指定目錄 // 3. 如果自己創建的so不需要重新編譯,可以將(app/build/intermediates/transforms)生成的so拷貝到這個目錄 jniLibs.srcDirs = ['libs'] // 如果是單個文件夾 可以直接這樣如下配置 // jniLibs.srcDir 'libs' } }}
此外,要再 gradle.properties 文件中添加android.useDeprecatedNdk=true,重新編譯工程即可。
不同CPU架構的Android手機加載時會在libs下找自己對應的目錄,從對應的目錄下尋找需要的.so文件;如果沒有對應的目錄,就會去armeabi下去尋找,如果已經有對應的目錄,但是如果沒有找到對應的.so文件,也不會去armeabi下去尋找了。 所以,這里需要注意工程配置哪幾個so文件目錄,需要加載對應的so文件,不然會報錯。
如何適配各個目錄,例如有一些第三方的類庫只提供了armeabi下的.so文件,而工程配置不止armeabi一個目錄,這就需要將armeabi下的.so文件復制到其他對應的目錄下。果第三方提供了不同平台的.so文件,則復制不同平台的.so文件到項目中對應的文件夾下即可。
so文件也會影響編譯出的apk大小(將apk解壓出來,lib目錄下就為so文件目錄),所以只配置armeabi一個目錄,既能適配各CPU架構的手機,也能精簡apk大小(微信就是只有armeabi一個目錄)。