溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

android多cpu架構(gòu)適配開篇

發(fā)布時(shí)間:2020-07-12 01:53:31 來源:網(wǎng)絡(luò) 閱讀:12754 作者:一劍圍城 欄目:移動(dòng)開發(fā)

簡介:做項(xiàng)目的時(shí)候經(jīng)常會(huì)使用到so文件。例如使用高德地圖,其SDK中就包含了armeabi、armeabi-v7a、arm64-v8a、x86等其他文件夾,里面通常放著同樣名稱、同樣數(shù)量的so文件。實(shí)際使用過程中,關(guān)于這些so文件引發(fā)的問題確實(shí)不少,也不好解決。寫下此文,希望以后遇到相關(guān)的問題,能有個(gè)大概的思路。


名詞解析:

    NDK:Native Development Kit

    JNI:Java Native Interface

    ABI: Application Binary Interface 應(yīng)用二進(jìn)制接口


Android Studio使用so庫

1、使用和eclipse一樣在libs目錄下新建armeabi目錄的方式

需要在build.gradle中添加指定jni庫目錄的語句

sourceSets {

   main.jniLibs.srcDirs = ['libs'] //指定libs為jni的存放目錄

}

2、使用AS默認(rèn)的位置:src/main/jniLibs

直接在src/main/下新建jniLibs目錄,將armeabi等目錄放到該目錄下即可

備注:AS可以直接右鍵新建同目錄下的jniLibs目錄,但該目錄不是編譯好的庫文件目錄,而是未編譯的本地代碼文件的目錄(這里指的是與java同級(jí)的jni目錄,放置cpp代碼的)


android支持的cpu架構(gòu)(目前是七種)


armeabi第5代 ARM v5TE,使用軟件浮點(diǎn)運(yùn)算,兼容所有ARM設(shè)備,通用性強(qiáng),速度慢
armeabi-v7a第7代 ARM v7,使用硬件浮點(diǎn)運(yùn)算,具有高級(jí)擴(kuò)展功能
arm64-v8a第8代,64位,包含AArch42、AArch74兩個(gè)執(zhí)行狀態(tài)對(duì)應(yīng)32、64bit
x86intel 32位,一般用于平板
x86_64intel 64位,一般用于平板
mips少接觸
mips64少接觸


安裝時(shí)的兼容性檢查:

   安裝到系統(tǒng)中后,so文件會(huì)被提取在:data/app/com.xxxxxxxx.app-x/lib/目錄下(5.0版本)、/data/app-lib/目錄下(4.2版本),其中armeabi和armeabi-v7a會(huì)生成arm目錄,arm64-v8a會(huì)生成arm64目錄。

   安裝app的時(shí)候,如果app使用了so文件,而不存在適合本機(jī)cpu架構(gòu)的so文件,會(huì)報(bào)如下錯(cuò)誤:

Installation failed with message INSTALL_FAILED_NO_MATCHING_ABIS.

例如:在x86模擬器上就必須有x86版本的so文件夾。不然無法安裝成功。


運(yùn)行時(shí)的兼容性檢查:

1、檢查目標(biāo)目錄下是否存在的so庫文件

2、檢查存在的so文件是否符合當(dāng)前cpu架構(gòu)。

   對(duì)于情況一,一般規(guī)避的做法是:保證jnilibs目錄下x86、x84_64、armeabi、armeabi-v7a、arm64-v8a等目錄下的文件名稱數(shù)量是一致的。

   例如:項(xiàng)目中使用了A、B、C三個(gè)第三方庫。其中A、B提供了armebi以及arm64-v8a版本的庫文件,而C只提供了armebi、armebi-v7a版本的庫文件。這時(shí)候只能夠刪除原有的arm64-v8a目錄,保留armeabi目錄,一般arm64的手機(jī)都能兼容使用armeabi版本的庫?;蛘邚?fù)制一份armeabi的so文件到缺少的目錄中(推薦)。


生成so文件:

   NDK交叉編譯時(shí)選定APP_ABI := armeabi x86 ...可以生成支持相應(yīng)芯片的so文件。APP_ABI := all生成支持所有芯片指令集(目前七種)so文件。


Android加載so文件規(guī)則:

   當(dāng)你只提供了armeabi目錄時(shí),armeabi-v7a、arm64-v8a架構(gòu)的程序都會(huì)去armeabi里尋找,而當(dāng)你同時(shí)也提供了armeabi-v7a、armeabi-v8a目錄,而里面又不存在對(duì)應(yīng)的so庫時(shí),系統(tǒng)就不會(huì)再去armeabi里面尋找了,直接找不到報(bào)錯(cuò)。其他平臺(tái)也是如此。這里我踩了不少的坑,切記。

   一般來說,一些比較有名的第三方庫都會(huì)提供armeabi、armeabi-v7a、x86這三種類型的so文件,同時(shí)擁有這三種版本的app可以在所有機(jī)型上運(yùn)行。另外,越來越多的SDK會(huì)同時(shí)提供arm64-v8a版本。只包含armeabi的項(xiàng)目也可以在所有設(shè)備上運(yùn)行。


現(xiàn)實(shí)案例:

   我的項(xiàng)目中使用了armeabi、arm64-v8a兩種類型,而當(dāng)我需要使用某語音第三方庫的時(shí)候,發(fā)現(xiàn)只提供了armeabi、armeabi-v7a兩種類型的so文件,而我的手機(jī)是arm64-v8a的。所以只會(huì)使用arm64-v8a里面的so文件,當(dāng)使用到該語音庫時(shí)找不到對(duì)應(yīng)的so庫,就會(huì)報(bào)錯(cuò)。理論上有以下兩種解決方法:

一、刪除所有arm64-v8a,只保留armeabi,全部使用兼容性最高的版本,但也運(yùn)行速度最慢。

二、將該語音庫的armeabi版本的so復(fù)制到arm64-v8a中。單一so文件使用armeabi兼容版本。


總結(jié):

  1. 當(dāng)你使用到so文件時(shí),保證每個(gè)子文件夾中文件名稱數(shù)量都是一致的。

  2. 對(duì)于只提供armeabi的第三方庫,復(fù)制一份armeabi的so文件到缺失的其他目錄中;或者只保留armeabi目錄(不推薦)


參考文檔:

   與 .so 有關(guān)的一個(gè)長年大坑

   Android 關(guān)于arm64-v8a、armeabi-v7a、armeabi、x86下的so文件兼容問題

   關(guān)于Android的.so文件你所需要知道的


向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI