Android游戲開發實踐(1)之NDK與JNI開發02

来源:http://www.cnblogs.com/alphagl/archive/2016/12/05/6132050.html
-Advertisement-
Play Games

Android游戲開發實踐(1)之NDK與JNI開發02 承接上篇 "Android游戲開發實踐(1)之NDK與JNI開發01" 分享完JNI的基礎和簡要開發流程之後,再來分享下在Android環境下的JNI的開發,以及涉及到的NDK相關的操作。當然,本篇仍是以Eclipse作為開發IDE,雖然Go ...


Android游戲開發實踐(1)之NDK與JNI開發02

承接上篇Android游戲開發實踐(1)之NDK與JNI開發01分享完JNI的基礎和簡要開發流程之後,再來分享下在Android環境下的JNI的開發,以及涉及到的NDK相關的操作。當然,本篇仍是以Eclipse作為開發IDE,雖然Google官方已經不再支持Eclipse了,推薦是用AndroidStudio進行開發。但對於游戲開發來說,IDE的影響並沒有那麼大,且從Eclipse那個時代過來的,對Eclipse還是感情很深的。後續,還有專門一篇來分享下AndroidStudio的使用以及使用CMake編譯等,會提到JNI這方面的內容。

作者:AlphaGL。版權所有,歡迎保留原文鏈接進行轉載 :)

按照慣例,每一篇文章都喜歡附上官方的文檔。因為,只有官方的文檔才是最準確,最實時,且內容最豐富的。那麼,NDK官方開發地址為:

Getting Started with the NDK:
https://developer.android.com/ndk/guides/index.html

本文目錄如下:

1、NDK環境搭建

(1)安裝CDT
CDT全成是C/C++ DevelopmentTools,安裝該插件使得Eclipse也得支持C/C++的開發。須下載和Eclipse版本對應的CDT插件。是喜歡Eclipse的便捷,同時又開發C/C++的必裝插件。CDT的下載地址為:
http://www.eclipse.org/cdt/downloads.php
安裝成功後,在Eclipse的Window-Preferences中看到多了一項C/C++的支持:

(2)NDK的下載
目前,NDK的最新版本為android-ndk-r13b,下載地址為:
https://developer.android.com/ndk/downloads/index.html
這裡需要說明下,為了方便演示筆者所使用的NDK版本為android-ndk-r8b。最新版本已經不再支持GCC編譯,預設改用Clang。還修複了相關的bug,建議線上的產品更新最新的穩定版。

(3)NDK的集成
將下載好的NDK解壓,並將該路徑添加到Path環境變數中,然後集成至Eclipse中。如圖:

2、交叉編譯

NDK編譯的環境有很多,但基本都是通過ndk-build工具來完成的。有直接通過Eclipse安裝CDT即可完編譯,也可以通過安裝Cygwin來編譯。事實上,在android-ndk-r7之後的版本,已經不需要安裝Cygwin就可以編譯出.so了。但這裡還是想介紹下,因為筆者開發曾用Cygwin編譯過一段時間,而且多瞭解一種編譯途徑也沒什麼壞處。當然,熟悉Linux平臺或者Mac平臺開發的朋友會感覺更親切些。

2.1 Cygwin編譯

Cygwin是一套在Window上模擬類Unix系統環境的工具。而Android底層又是基於Linux的。因此,對Linux環境下的開發支持也更好。只要在Cygwin中安裝gccg++gdbmakeGUN工具集即可。
(1)Cygwin的安裝
Cygwin的下載地址為:
https://cygwin.com/install.html

(2)Cygwin的安裝步驟:
下載完setup-x86_64.exe,直接下一步:

這裡有三個選項,分別是從網路安裝,只下載不安裝,從本地目錄安裝(如果,之前安裝過)。可根據自己的實際情況選擇。這裡選擇從網路安裝。然後,下一步

這裡選擇第一項,Direct Connection。然後,下一步

這裡下載地址選擇mirrors.kernel.org即可。也可選擇國內163的鏡像地址。

這裡選擇要安裝的包(autoconfautomakemakegccg++gdbpcregawk等)。這裡偷懶就直接把AdminDebugDevelDocEditorsShells,當然還有Python。然後,下一步

接著經過漫長的等待,大概下載3,4G的文件。

安裝完,運行Cygwin,輸入如下命令

(3)在Cygwin下配置NDK環境變數
在當前當前用戶目錄下運行,(例如,我的目錄為D:\env\cygwin\home\John):

配置NDK環境變數,註意別覆蓋原來的PATH環境變數。

(4)驗證NDK配置
嘗試在Cygwin下用ndk-build來編譯NDK下的hello-jni的samples。如圖:

可以看到正確編譯出libhello-jni.so庫(在項目目錄下的libs,不同cup架構命名的文件夾里)。如遇到各種許可權錯誤,請將samples下的hello-jni項目的許可權修改為可寫入、可修改等。

2.2 Eclipse編譯

(1)將hello-jni的項目導入到Eclipse中。


(2)給hello-jni添加builder,來編譯C/C++代碼。
右鍵HelloJni項目,選擇Properties,然後,選擇Builders,點擊New新建一個Builder。選擇Program,點擊OK即可。

分別填寫Builder名稱。找到Cygwin的Shell程式和工作目錄。將要編譯的項目目錄和執行的命令當成參數參數Shell命令執行。

註意: cd/cygdrive/d/android-ndk-r8b/samples/hello-jni中間有個空格。
ANDROID_NDK_ROOT:是在Cygwin中配置NDK環境變數的名稱。

通過Cygwin中輸入bash --login -h可以獲取更詳細的信息:

Your group is currently "mkpasswd".  This indicates that your
gid is not in /etc/group and your uid is not in /etc/passwd.

The /etc/passwd (and possibly /etc/group) files should be rebuilt.
See the man pages for mkpasswd and mkgroup then, for example, run

mkpasswd -l [-d] >> /etc/passwd
mkgroup  -l [-d] >> /etc/group

如果遇到這種,按照提示在Cygwin終端執行,mkpasswd -l [-d] >> /etc/passwdmkgroup -l [-d] >> /etc/group命令即可。

(3)將配置JNI_Builder優先順序設為最高

(4)編譯HelloJni工程。
選中HelloJni工程,在Eclipse中選擇Project-clean。這樣,Eclipse便可自動編譯HelloJni工程了。

2.3 AndroidStudio和CMake編譯

這裡就不花篇幅介紹這相關的內容,下一篇專門介紹下AndroidStudio的使用及在AndroidStudio下NDK的開發。希望能給從其它IDE遷移到IntelliJ IDEA系開發或許剛接觸AndroidStudio一些啟發。所以,這裡先留個伏筆。

ndk-build、Android.mk與Application.mk

雖然,已經成功的將samples下的hell-jni項目成功編譯出了.so動態庫。在整個交叉編譯過程中,涉及到了三個比較重要的文件,分別是ndk-buildAndroid.mkApplication.mk,所以,有必要瞭解一下,這三個文件在整個交叉編譯過程中起了什麼作用。

3、ndk-build

首先,ndk-build是一個shell腳本,目標是幫助你正確的調用NDK的構建腳本。ndk-build<ndk-root-path>(NDK安裝目錄根路徑下)有個ndk-build的shell腳本文件,或ndk-build.cmd的文件。

ndk-build的官方指南為:

https://developer.android.com/ndk/guides/ndk-build.html

3.1 ndk-build用法
cd $PROJECT_PATH
$ <ndk>/ndk-build

用法:在項目的根目錄下,執行ndk-build腳本命令。

例如:

進到hello-jni的項目根目錄執行ndk-build,ANDROID_NDK_ROOT是在Cygwin中配置NDK環境變數的名稱。

$ <ndk>/ndk-build -C <PROJECT_PATH>

用法:在任意目錄下執行ndk-build,用-C來指定要編譯的項目的目錄。

例如:

實際上:執行ndk-build相當於執行了以下命令:

$GNUMAKE -f <ndk>/build/core/build-local.mk
<parameters>

例如:

3.2 ndk-build可選參數
$ ndk-build clean
清除編譯生成的二進位文件。

$ ndk-build -C <project>
指定項目路徑

$ ndk-build NDK_DEBUG=0
NDK_DEBUG為0是編譯為release版,為debug版。

更多的ndk-build的參數介紹,請參考上面貼出的ndk-build的官方指南。

4、Android.mk

Android.mk是用來向編譯系統指定項目中C/C++源代碼文件編譯、鏈接規則的一種描述文件。它是GUN makefile文件的一小部分。那麼,簡單來說,就是用來起指定編譯引用的頭文件目錄、編譯出的so的名字、需要編譯的源文件或庫等作用。熟悉makefile語法的,肯定熟悉這種用法。Android.mk文件在$project-path/jni/Android.mk路徑下。

Android.mk官方說明文檔地址為:
https://developer.android.com/ndk/guides/android_mk.html

4.1 Android.mk初級

首先,仍以hello-jni為例,看看都定義了哪些內容。打開<ndk-path>/samples/hello-jni/jni/Android.mk文件。

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE    := hello-jni
LOCAL_SRC_FILES := hello-jni.c

include $(BUILD_SHARED_LIBRARY)

說明:
LOCAL_PATH := $(call my-dir)
Android.mk必須首先定義LOCAL_PATH,它用來在開發的樹文件夾中定位文件的。my-dir是由編譯系統提供的巨集,用來返回當前目錄的路徑。(註意:這個路徑是包含了Android.mk的路徑)

include $(CLEAR_VARS)
CLEAR_VARS變數也是由編譯系統提供的,include $(CLEAR_VARS)是引用一個特殊的GUN makefile文件,這個makefile文件所做的就是清除定義了很多LOCAL_XXX(例如: LOCAL_MODULE,LOCAL_SRC_FILES,LOCAL_STATIC_LIBRARIES等)這種格式的變數,但這裡不會清除LOCAL_PATH變數。這麼做是很有必要的,因為編譯系統解析這些編譯控制文件都是在單一的GUN make上下文環境中,解析出來的LOCAL_XXX變數都是全局的。

LOCAL_MODULE := hello-jni
LOCAL_MODULE必須是唯一的,且不能包含空格。編譯系統會根據這個名字生成相應的共用庫,並自動添加首碼和尾碼。本例中,最終生成的共用庫的名稱為libhello-jni.so

註意:編譯生成的共用庫都是lib開頭的,如果,你聲明的名稱已經包含lib(libhello-jni),那麼,最終生成的共用庫名稱就不添加lib首碼了,生成的仍為libhello-jni.so

LOCAL_SRC_FILES := hello-jni.c
LOCAL_SRC_FILES用來指明要編進共用庫(.so)的C/C++源文件的列表。

註意:這裡只需要指定要編譯的.c或者.cpp等源文件即可,不需要指定.h頭文件。

include $(BUILD_SHARED_LIBRARY)
BUILD_SHARED_LIBRARY變數是由系統提供,include $(BUILD_SHARED_LIBRARY)會引用一個Gun makefile腳本,用來收集你定義的LOCAL_XXX格式的變數。同時,決定哪些要編譯,如何編譯等。

註意:顯然BUILD_SHARED_LIBRARY是用來編譯出共用庫.so文件的,同理,也可以使用BUILD_STATIC_LIBRARY來編譯出靜態庫.a文件。

以上便是編寫一個簡單的Android.mk文件的所有元素。通過上面的描述發現,如果我們要編寫一個自己的Android.mk文件,沒有特殊需求的話,可以直接將hello-jni工程中的Android.mk文件拷貝,然後,修改庫的名稱(LOCAL_MODULE)和要編譯的源文件列表(LOCAL_SRC_FILES)變數即可。

4.2 Android.mk高級

這裡來詳細瞭解下Android.mk其他的一些變數及語法規則。

(1)NDK變數與巨集
Android.mk中還有一些其他變數,是作為NDK編譯系統的保留變數,你只能依賴它或者定義它。這些變數的規則如下:

  • LOCAL_開頭的變數名稱(如:LOCAL_MODULELOCAL_PATH等)
  • PRIVATE_NDK_APP開頭的變數名稱(編譯系統內部使用)
  • 小寫的名稱(例如:my-dir,同樣,也是作為內部使用)

如果你需要在Android.mk中定義自己的變數,推薦用MY_作為首碼。

(2)NDK定義的變數
CLEAR_VARS
上面已經介紹過了,這裡就不在贅述。記住一點,在定義LOCAL_XXX前,必須引用這個腳本。用法:

include $(CLEAR_VARS)

BUILD_SHARED_LIBRARY
該變數指向了一個腳本,這個腳本會收集你在每個模塊定義的LOCAL_XXX變數信息,並且這個變數還確定了怎樣使用你的源碼去編譯一個共用庫。註意,使用這個變數需要你至少已經定義了LOCAL_MODULELOCAL_SRC_FILES。該變數會使編譯系統生成一個以.so結尾的庫。用法:

include $(BUILD_SHARED_LIBRARY)

BUILD_STATIC_LIBRARY
該變數是BUILD_SHARED_LIBRARY的一個變體,是用來生成一個靜態庫。構建系統並不會把靜態庫包含進你的工程裡面,但是可以利用靜態庫生成共用庫。該變數會使編譯系統生成一個以.a結尾的庫。

include $(BUILD_STATIC_LIBRARY)

PREBUILT_SHARED_LIBRARY
指向一個腳本,這個腳本被用來指定一個預構建的共用庫。與BUILD_SHARED_LIBRARYBUILD_STATIC_LIBRARY不同,LOCAL_SRC_FILES的值不能是一個源文件,它必須是一個單獨的指向預構建的共用庫的路徑,例如:foo/libfoo.so。用法:

PREBUILT_STATIC_LIBRARY
該變數與PREBUILT_SHARED_LIBRARY相同,只是指向的一個預構建的靜態庫。

TARGET_ARCH
這個變數是目標CPU架構的名字,就像Android Open Source Project裡面指定了目標CPU架構。這個變數用於任意的ARM相容的構建,或者ARM,或者是獨立於CPU結構的修訂,或者ABI。

TARGET_PLATFORM
編譯到目標平臺的api等級。例如,Android5.1對應的是Android api22。用法:

TARGET_PLATFORM := android-22

TARGET_ARCH_ABI
當編譯系統解析Android.mk文件的時候,這個變數存儲CPU和架構的名字。你可以指定一個或者多個下麵列出的名字,使用空格分隔兩個名字

用法:

TARGET_ARCH_ABI := arm64-v8a

註意:android-ndk-1.6_r1之前這個這個變數被定義為arm。

TARGET_ABI
該變數將API的級別和ABI聯繫在一起,當你在真機上調試系統的時候特別有用。用法:

TARGET_ABI := android-22-arm64-v8a

LOCAL_MODULE_FILENAME
這是一個可選變數。允許你重新指定一個變數的名稱來覆蓋預設生成的名稱。例如:強制生成libnewfoo.so

LOCAL_MODULE := foo
LOCAL_MODULE_FILENAME := libnewfoo

LOCAL_MODULE_FILENAME不需要指定文件路徑或擴展名

LOCAL_SRC_FILES:
該變數用來指定要編譯的源文件列表。這裡推薦使用相對路徑。用法:

LOCAL_SRC_FILES := foo1.c \
../Module2/foo2.c

註意:使用Unix風格的/,多個文件使用\換行,註意\後面沒有空格。

LOCAL_CPP_EXTENSION
同樣,LOCAL_CPP_EXTENSION也是一個可選變數。用來指定C++源文件的擴展名。預設是.cpp。從NDK r7版本後,可以指定一系列的擴展名。用法:

LOCAL_CPP_EXTENSION := .cxx .cpp .cc

LOCAL_CPP_FEATURES
同樣,LOCAL_CPP_FEATURES也是一個可選變數。如果,你用到了C++的一些特殊功能(例如:RTTI,異常支持等),並且正確的編譯和鏈接,可以使用該變數來聲明。用法:

LOCAL_CPP_FEATURES := rtti exceptions

建議使用該變數來代替LOCAL_CPPFLAGS中直接定義-frtti-fexceptions這種用法

LOCAL_C_INCLUDES
可選變數,可以通過該變數來指定一個相對於NDK根目錄的路徑列表,併在編譯C/C++時添加到搜索路徑中。用法:

LOCAL_C_INCLUDES := $(LOCAL_PATH)//foo

註意:該聲明要放在LOCAL_CFLAGSLOCAL_CPPFLAGS的聲明前面。

LOCAL_CFLAGS
可選變數,在編譯C/C++源代碼時,能給編譯器傳遞一個編譯標誌集合。用來指定附加巨集的定義和編譯選項是很有用的。

LOCAL_CPPFLAGS
可選變數,在編譯C++源代碼時(只編譯C++),能給編譯器傳遞一個編譯標誌集合。

LOCAL_STATIC_LIBRARIES:
該變數定義了本模塊編譯鏈接過程中用到的靜態庫列表。

LOCAL_SHARED_LIBRARIES
該變數定義了本模塊編譯鏈接過程中用到的共用庫列表。

LOCAL_WHOLE_STATIC_LIBRARIES
該變數跟LOCAL_STATIC_LIBRARIES類似,不同的是在編譯鏈接過程中會載入靜態庫的所有源代碼目標文件。這在解決幾個庫之間迴圈引用時,非常有用。可以通過GUN的--whole-archive標誌來查看相關說明。

LOCAL_LDLIBS
該變數用來定義本模塊編譯時用到的附加的鏈接器選項。用-l首碼指定。例如:要鏈接/system/lib/libz.so

LOCAL_LDLIBS := -lz

註意:如果,你編譯一個靜態庫是定義了該變數,編譯系統會忽略它,並且ndk-build會列印一個警告。

LOCAL_LDFLAGS
該變數定義了在編譯給編譯系統傳遞一些其他的鏈接標誌。用法:

LOCAL_LDFLAGS += -fuse-ld=bfd

註意:如果,你編譯一個靜態庫是定義了該變數,編譯系統會忽略它,並且ndk-build會列印一個警告。

LOCAL_ALLOW_UNDEFINED_SYMBOLS
預設情況下,在編譯一個共用庫的時候,任何未定義的引用,將會拋出"符號未定義(undefined symbol)"的錯誤。該變數能幫助你捕捉代碼中的bug。如果,要禁用這項檢查可以把該變數設置為true。這麼設置

註意:如果,你編譯一個靜態庫是定義了該變數,編譯系統會忽略它,並且ndk-build會列印一個警告。

LOCAL_ARM_MODE
預設情況下,編譯系統會在ARM平臺"thumb"模式下生成16位的二進位文件。定義該變數則強制生成32位"arm"模式下的對象文件。例如:

LOCAL_ARM_MODE := arm

你也可以加上.arm尾碼來告訴編譯系統,只想對某個源文件使用arm指令。例如:

LOCAL_SRC_FILES := foo.c bar.c.arm

LOCAL_ARM_NEON
只有當目標平臺為armeabi-v7a指令集時,才定義它。它允許你的C/C++代碼中使用ARM高級指令。也可以在彙編文件中使用NEON指令。你可以使用.neon尾碼來指定編譯器支持NEON指令編譯。例如:

LOCAL_SRC_FILES = foo.c.neon bar.c zoo.c.arm.neon

註意:不是所有的ARMv7架構的CPU都支持NEON擴展。

LOCAL_DISABLE_NO_EXECUTE
Android NDK r4版本開始支持這種"NX bit"的安全功能。預設是啟用的,你也可以設置該變數的值為true來禁用它。但不推薦這麼做。該功能不會修改ABI,只在ARMv6+CPU的設備內核上啟用。

LOCAL_DISABLE_RELRO
預設情況下,NDK編譯代碼是只讀重定位和GOT保護的。這個會指示運行時鏈接器標記特定的記憶體區是只讀的,在移動位置之後。這樣會使得某些安全漏洞(如GOT覆蓋)更難執行。預設是啟用的,你也可以把該變數的值設為true來禁用。但不推薦這麼做。

LOCAL_DISABLE_FORMAT_STRING_CHECKS
預設情況下,編譯系統編譯代碼時會檢查格式化字元串,如果printf樣式的函數使用了非常嚴格的字元串,那麼編譯出錯。預設是開啟的,你也可以通過設置該變數的值為true來禁用。但不推薦這麼做。

LOCAL_EXPORT_CFLAGS
該變數是用來記錄C/C++編譯器標誌集合,這些編譯器標誌會被添加到通過變數LOCAL_STATIC_LIBRARIESLOCAL_SHARED_LIBRARIES使用本模塊的其他模塊的LOCAL_CFLAGS變數中。例如:

include $(CLEAR_VARS)
LOCAL_MODULE := foo
LOCAL_SRC_FILES := foo/foo.c
LOCAL_EXPORT_CFLAGS := -DFOO=1
include $(BUILD_STATIC_LIBRARY)


include $(CLEAR_VARS)
LOCAL_MODULE := bar
LOCAL_SRC_FILES := bar.c
LOCAL_CFLAGS := -DBAR=2
LOCAL_STATIC_LIBRARIES := foo
include $(BUILD_SHARED_LIBRARY)

編譯bar模塊時,"-DFOO=1 -DBAR=2"標誌將一起傳遞給編譯器。

LOCAL_EXPORT_CPPFLAGS
該變數與LOCAL_EXPORT_CFLAGS類似,但只適用於C++。

LOCAL_EXPORT_C_INCLUDES
該變數與LOCAL_EXPORT_CFLAGS類似,但是該變數用來包含路徑的。例如,上例中bar.c需要包含foo模塊的頭文件。

LOCAL_EXPORT_LDFLAGS:
該變數與LOCAL_EXPORT_CFLAGS類似,但是它用作鏈接器標誌。

LOCAL_EXPORT_LDLIBS
該變數與LOCAL_EXPORT_CFLAGS一樣,該變數的值將會被添加到其它模塊引用到本模塊的其它模塊的LOCAL_LDLIBS變數中。例如:

include $(CLEAR_VARS)
LOCAL_MODULE := foo
LOCAL_SRC_FILES := foo/foo.c
LOCAL_EXPORT_LDLIBS := -llog
include $(BUILD_STATIC_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := bar
LOCAL_SRC_FILES := bar.c
LOCAL_STATIC_LIBRARIES := foo
include $(BUILD_SHARED_LIBRARY)

編譯bar的時候,會在鏈接log的系統日誌庫。

LOCAL_SHORT_COMMANDS
當你的模塊有很多源代碼文件或依賴很多靜態庫或者共用庫時,設置為true。這樣會強制編譯系統使用@語法來包含中間對象或鏈接庫來生成歸檔文件

註意:這個功能在Windows上很有用,因為Windows上的命令行支持的最大字元數為8191個,這對於複雜的項目來說太小。預設是不推薦啟用這個功能,因為它會使編譯變得很慢。

LOCAL_THIN_ARCHIVE
編譯靜態庫是,如果該變數值設為true,會生成一個較小的歸檔文件。並不包含目標文件,而是用目標文件的路徑替代。有效值是truefalse和空。

註意:如果該模塊不是編譯為靜態塊,或者預編譯靜態庫,該值將被忽略。

LOCAL_FILTER_ASM
該變數的值將作為一個Shell命令,它會過濾從LOCAL_SRC_FILES生成的文件或彙編文件。定義該變數將會發生下麵的情況:

  • 編譯系統會將C/C++源文件生成臨時的彙編文件,而不是將他們編譯到目標文件中。
  • 編譯系統會對彙編文件和LOCAL_SRC_FILES中列出的文件執行LOCAL_FILTER_ASM中的Shell命令,生成另外一個彙編文件。
  • 編譯系統將這些過濾後的彙編文件編譯進目標文件。

NDK提供的函數巨集
NDK提供了GNU Make的函數巨集。使用$(call <function>)的方式調用,它們會返回相應的文本信息。
my-dir
該巨集返回的是最後包含的makefile文件路徑,一般是當前Android.mk的路徑。my-dir對於Android.mk開頭定義的LOCAL_PATH變數很有用。例如:

LOCAL_PATH := $(call my-dir)

由於GNU Make的工作方式,這個巨集返回的是構建系統在解析構建腳本時包含的最後一個makefile的路徑。因此,你不應該在include其他的文件之後再繼續使用my-dir。例如:

LOCAL_PATH := $(call my-dir)

# ... declare one module

include $(LOCAL_PATH)/foo/`Android.mk`

LOCAL_PATH := $(call my-dir)

# ... declare another module

這裡的問題在於第二個my-dir的調用將LOCAL_PATH的值設置為了$PATH/foo,因為$PATH/foo才是最近包含的路徑。你可以通過在Android.mk文件中放置額外的包含來避免這個問題。例如:

LOCAL_PATH := $(call my-dir)

# ... declare one module

LOCAL_PATH := $(call my-dir)

# ... declare another module

# extra includes at the end of the Android.mk file
include $(LOCAL_PATH)/foo/Android.mk

如果這種方式不可行,那麼可以將第一次調用my-dir的值存在另外一個變數裡面,例如:

MY_LOCAL_PATH := $(call my-dir)

LOCAL_PATH := $(MY_LOCAL_PATH)

# ... declare one module

include $(LOCAL_PATH)/foo/`Android.mk`

LOCAL_PATH := $(MY_LOCAL_PATH)

# ... declare another module

all-subdir-makefiles
該巨集返回的是當前my-dir路徑下的所有子目錄中的Android.mk文件的列表。你可以使用此函數向構建系統提供深層嵌套的源目錄層次結構。預設情況下,NDK僅查找包含Android.mk文件的目錄中的文件。

this-makefile
該巨集返回的是當前makefile文件的路徑(從構建系統調用這個函數中)。

parent-makefile
該巨集返回當前目錄樹中的父makefile的路徑(包含當前makefile的makefile路徑)。

grand-parent-makefile
該巨集返回包含樹中的祖父類makefile的路徑(包含當前makefile的makefile的路徑)。

import-module
該巨集允許你通過模塊的名稱找到並包含模塊的Android.mk文件。例如:

$(call import-module,<name>)

在這個示例中,構建系統會根據NDK_MODULE_PATH這個環境變數所指示的目錄裡面尋找名為<name>的模塊,然後自動為你include對應的Android.mk文件。

5、Application.mk

通過上面的介紹,大體瞭解了Android.mk文件的用法及規則。但通常編譯本地C/C++代碼光有Android.mk還不夠,還得需要Application.mk文件。Application.mk也是一種makefile文件,跟Android.mk有相似之處。如果說,Android.mk用來描述單獨某個模塊的編譯規則的描述文件,那麼Application.mk則是描述整個應用程式的模塊的描述文件。Application.mk文件一般在$project-path/jni/Application.mk下($project-path是項目根目錄)。當然,也可以放在$NDK/apps/<myapp>/Application.mk路徑下。這兩種方式,造成Application.mk也有細微的區別。

(1)變數
APP_PROJECT_PATH
該變數用來指定項目根目錄的絕對路徑。

註意:假如Application.mk文件的路徑是$NDK/apps/<myapp>/Application.mk,那麼該變數為強制定義的。如果,Application.mk文件在$project-path/jni/Application.mk路徑下,則是可選變數。

APP_OPTIM:
該變數為可選變數,值為releasedebug。編譯應用程式模塊的時,可以用來改變優化級別。預設是release模式,並且會生成高度優化的二進位文件。debug模式生成的是未優化的二進位代碼,但更容易調試。

APP_CFLAGS
在編譯任何模塊的任何C/C++代碼時,構建系統會通過該變數給編譯器傳遞一個C編譯標誌集合。你可以使用該變數根據應用程式中給定的模塊的需要來改變其構建,而不需要修改Android.mk文件本身了。
這些標誌的所有路徑必須相對於NDK的頂級目錄。例如:如果你有以下設置:

sources/foo/Android.mk
sources/bar/Android.mk

在編譯期間,你需要在foo/Android.mk中指定你要添加的foo的源路徑,你應該使用:

APP_CFLAGS += -Isources/bar

或者:

APP_CFLAGS += -I$(LOCAL_PATH)/../bar

使用-I../bar將不能正常工作,因為它等價於-I$NDK_ROOT/../bar

註意:在android-ndk-1.5_r1版本中,該變數只適用於C,不能作用於C++。之後的所有的版本,該變數可適用C/C++源碼上。

APP_CPPFLAGS
該變數包含一組C++編譯器標誌,構建系統僅在構建C++源代碼時傳遞給編譯器。

APP_LDFLAGS
在鏈接的時候,構建系統系統會想鏈接器傳遞一組鏈接標誌。該變數僅在構建系統構建共用庫和可執行文件的時候才生效,當構建靜態庫時,將被忽略。

APP_BUILD_SCRIPT
預設情況下,NDK構建系統會在$project-path/jni/目錄下查找Android.mk文件。如果你想修改此行為,你可以定義APP_BUILD_SCRIPT變數,並指向備用的構建腳本。編譯系統總是將一個非絕對路徑解釋為NDK的頂級目錄。

APP_ABI
預設情況下,編譯系統會根據armeabi ABI生成機器碼。該機器碼基於ARMv5TE並且支持浮點運算的CPU。你可以使用APP_ABI參數來指定不同的ABI。不同的指令集的APP_ABI設置如下:

註意:all是android-ndk-r7版本開始支持的。
你也可以指定多個值,將它們放在同一行,中間用空格隔開。例如:

APP_ABI := armeabi armeabi-v7a x86 mips

APP_PLATFORM
此變數包含目標Android的名稱。例如:android-3對應的是Android1.5的系統鏡像。

APP_STL
預設情況下,NDK構建系統只為最小的C++運行庫(/system/lib/libstdc++.so)提供C++頭文件。此外,你還可以在自己的應用程式中使用或鏈接其他C++實現。可以使用APP_STL來選擇其中的一個。例如:

APP_STL := stlport_static
APP_STL := stlport_shared
APP_STL := system

APP_SHORT_COMMANDS
該變數相當於整個項目的Android.mk中定義了LOCAL_SHORT_COMMANDS

NDK_TOOLCHAIN_VERSION
將此變數定義為4.9版本的GCC編譯器。在android-ndk-r13預設是Clang編譯器。

APP_PIE
從Android 4.1(API級別16)開始,Android的動態鏈接器支持位置無關可執行文件(PIE)。從Android 5.0(API級別21),可執行文件需要PIE。要使用PIE構建可執行文件,需設置-fPIE標誌。這個標誌會使得通過隨機代碼的位置來查找記憶體損壞的bug更加困難。預設情況下,如果您的項目目標為Android-16或更高版本,ndk-build會自動將此值設置為true。您可以將其手動設置為true或false。

註意:此標誌僅適用於可執行文件。它在構建共用或靜態庫時沒有任何作用。

APP_THIN_ARCHIVE
相當於在Android.mk文件中為此項目中的所有靜態庫模塊設置LOCAL_THIN_ARCHIVE的預設值。

以上內容可能不一定完全正確,大部分內容是根據Android官方文檔,通過自己的理解轉述的。並沒有每個變數在.mk文件中有實踐過。所以,如果遇到你覺得有問題的地方,歡迎與我交流。

技術交流QQ群:528655025
作者:AlphaGL
出處:http://www.cnblogs.com/alphagl/
版權所有,歡迎保留原文鏈接進行轉載 :)



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 組內非同步會與組外順序執行的事件爭搶資源 1)、創建一個組 2)、組內非同步ST1,DISPATCH_QUEUE_PRIORITY_DEFAULT 為預設優先順序 3)、組內非同步ST2,DISPATCH_QUEUE_PRIORITY_DEFAULT 為預設優先順序 4)、組內通知,獲取主線程。組內非同步全部執 ...
  • viewport預備知識 dpr === dppx dpr:device pixel ratio 設備像素比 dppx:Number of dots per px unit 每像素有多少點 。 1dppx = 96dpi dpr = 設備物理像素 / 設備獨立像素 設備物理像素指顯示在移動端中實際的 ...
  • 在學習新內容之前,我們先來弄清楚兩個問題: 1 . 什麼是ViewGroup? ViewGroup是一種容器。它包含零個或以上的View及子View。 2 . ViewGroup有什麼作用? ViewGroup內部可以用來存放多個View控制項,並且根據自身的測量模式,來測量View子控制項,並且... ...
  • 提交審核後進去下麵鏈接申請加急審核鏈接:https://developer.apple.com/appstore/contact/appreviewteam/index.html 在I would like to選擇加急審核(預設選項) 填寫相關信息 最後一項可以不填 最後是申請原因跟理由 看見別人 ...
  • 我們在手機上查看相冊時,首先看到的是網格狀的圖片展示界面,然後我們選擇想要欣賞的照片點擊進入,這樣就可以全屏觀看該照片,並且可以通過左右滑動來切換照片。如下圖的顯示效果: 首先我們先羅列一下本次實現所要用的知識點: (1)首頁界面,我們需要可以通過GridView來實現,有關GridView的實現代 ...
  • 1、並行-非同步(ST1與ST2搶占資源) 1-1)、獲取 並行(全局)隊列 ,DISPATCH_QUEUE_PRIORITY_DEFAULT 為預設優先順序。 1-2)、創建非同步事件。 2、串列-同步(順序執行) 2-1)、創建串列隊列,DISPATCH_QUEUE_SERIAL 為串列 2-2)、創 ...
  • 前言 在學習一門新技術的你也許有跟我一樣的困惑,照著書上或者視頻上的敲了。但是就是有各種問題沒有出來自己想要的結果。我會將自己在這個過程中遇到的坑都記錄下來,不一定全覆蓋,但希望這些文章可以解決你的問題。 錯誤提示 Invariant Violation:Applicaction 項目名 has n ...
  • onClickLogin()方法: loginListener介面: updateUserInfo()方法: 必須重寫onActivityResult()方法 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...