溫馨提示×

溫馨提示×

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

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

Android NDK生成及連接靜態(tài)庫與動態(tài)庫的方法

發(fā)布時(shí)間:2020-09-15 06:06:13 來源:腳本之家 閱讀:373 作者:zenny_chen 欄目:移動開發(fā)

對于Android應(yīng)用開發(fā),大部分情況下我們使用Java就能完整地實(shí)現(xiàn)一個(gè)應(yīng)用。但是在某些情況下,我們需要借助C/C++來寫JNI本地代碼。比如,在使用跨平臺的第三方庫的時(shí)候;為了提升密集計(jì)算性能的時(shí)候(這種情況下往往還可能會直接使用匯編語言)。因此,這里我將為大家介紹如何給其它開發(fā)者創(chuàng)建可供使用的靜態(tài)庫或動態(tài)庫。而應(yīng)用開發(fā)者如何去連接這些生成的靜態(tài)庫或動態(tài)庫。由于現(xiàn)在Android Studio已經(jīng)比較成熟,因此以下描述將基于Android Studio的目錄布局。

在Android Studio的一個(gè)項(xiàng)目工程下,會有一個(gè)app文件夾,這個(gè)文件夾主要存放我們編寫應(yīng)用的所有代碼以及相關(guān)其它資源。如果我們需要寫JNI,那么就需要在這個(gè)目錄下創(chuàng)建jni文件夾。jni文件夾里必須包含Android.mk文件、Application.mk文件以及你所要編譯的源文件。下面我們先給出編譯生成提供給第三方開發(fā)者使用的靜態(tài)庫與動態(tài)庫的Android.mk文件內(nèi)容:

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

# 編譯后生成的模塊名,在Java端使用System.loadLibrary加載模塊的的時(shí)候直接用此名
LOCAL_MODULE := my_blocks

# LOCAL_SRC_FILES用于添加源文件(可以是.c,.cpp,.s等)
LOCAL_SRC_FILES := my_block_test.c
LOCAL_SRC_FILES += my_blocks_data.c
LOCAL_SRC_FILES += my_runtime.c

# 條件預(yù)編譯,如果當(dāng)前架構(gòu)為ARMv7以下的架構(gòu),則定義MY_NO_CSWAP這個(gè)宏
ifeq ($(TARGET_ARCH_ABI),armeabi)
  LOCAL_CFLAGS := -DMY_NO_CSWAP=1
endif

# 連接系統(tǒng)編譯出的靜態(tài)庫
LOCAL_STATIC_LIBRARIES := cpufeatures
LOCAL_STATIC_LIBRARIES += ld.gold

LOCAL_LDLIBS := -llog

# 用于生成動態(tài)庫
# include $(BUILD_SHARED_LIBRARY)

# 用于生成靜態(tài)庫
include $(BUILD_STATIC_LIBRARY)

$(call import-module,cpufeatures)

然后,我們把上述編譯腳本中所列出的my_block_test.c、my_blocks_data.c以及my_runtime.c這三個(gè)源文件放在jni目錄下即可編譯。

最后,用Android JNI編譯器工具——ndk_build編譯完之后就可在生成的obj目錄下看到my_blocks.a文件了。在Unix/Linux下,.a表示靜態(tài)庫文件;.so表示動態(tài)共享庫文件。

下面我們將創(chuàng)建另一個(gè)工程,這個(gè)工程將構(gòu)建最終可執(zhí)行的應(yīng)用。其Android.mk文件描述如下:

LOCAL_PATH := $(call my-dir)

### 以下這段用于預(yù)構(gòu)建我們將要連接的已存在的靜態(tài)庫或動態(tài)庫 ###

include $(CLEAR_VARS)

# 我們將連接已編譯好的my_blocks模塊
LOCAL_MODULE  := my_blocks

# 填寫源文件名的時(shí)候,要把靜態(tài)庫或動態(tài)庫的文件名填寫完整。
# $(TARGET_ARCH_ABI)/ 表示將不同架構(gòu)下的庫文件存放到相應(yīng)架構(gòu)目錄下
LOCAL_SRC_FILES := $(TARGET_ARCH_ABI)/libmy_blocks.a

# LOCAL_SRC_FILES := $(TARGET_ARCH_ABI)/libmy_blocks.so

# 用于預(yù)構(gòu)建靜態(tài)庫(后面可被連接)
include $(PREBUILT_STATIC_LIBRARY)

# 用于預(yù)構(gòu)建動態(tài)庫(后面可被連接)
# include $(PREBUILT_SHARED_LIBRARY)

# 這里要注意的是,對于一次預(yù)構(gòu)建只能預(yù)構(gòu)建動態(tài)庫,要么是靜態(tài)庫,兩者不能共存

### 以下內(nèi)容用于描述編譯當(dāng)前工程的源代碼 ###

include $(CLEAR_VARS)

LOCAL_MODULE := mytest

LOCAL_SRC_FILES := test.c

LOCAL_STATIC_LIBRARIES := cpufeatures
LOCAL_STATIC_LIBRARIES += ld.gold

# 連接我們前面聲明好的靜態(tài)庫
LOCAL_STATIC_LIBRARIES += my_blocks

# 連接我們前面聲明好的動態(tài)庫
# LOCAL_SHARED_LIBRARIES += my_blocks

LOCAL_LDLIBS := -llog

# 將此模塊構(gòu)建為動態(tài)庫
include $(BUILD_SHARED_LIBRARY)

$(call import-module,cpufeatures)

下面我們來看一下這個(gè)工程jni的目錄結(jié)構(gòu)內(nèi)容:

Android NDK生成及連接靜態(tài)庫與動態(tài)庫的方法

在arm64-v8a、armeabi等每個(gè)架構(gòu)名目錄下,都要包含有我們在第一個(gè)工程中生成出來的靜態(tài)庫或動態(tài)庫文件,并且要與架構(gòu)名相一致。比如一下是arm64-v8a下的內(nèi)容:

Android NDK生成及連接靜態(tài)庫與動態(tài)庫的方法

然后,我們需要編輯Application.mk文件,如下所示:

# 使用當(dāng)前NDK編譯器所支持的所有處理器架構(gòu)
APP_ABI := all

# 使用LLVM Clang 3.6編譯器工具鏈
NDK_TOOLCHAIN_VERSION=clang3.6

# 開啟C11標(biāo)準(zhǔn),外加GNU語法擴(kuò)展
APP_CFLAGS += -std=gnu11

# 啟用Blocks語法
APP_CFLAGS += -fblocks

最后,我們在Java端只需加載我們當(dāng)前所要執(zhí)行的動態(tài)庫模塊即可,不需要關(guān)心前一個(gè)工程所生成的、用于給當(dāng)前JNI模塊所使用的動態(tài)庫或靜態(tài)庫。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

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

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

AI