AndroidStudio項目CMakeLists解析

下面分別做介紹:

cmake_minimum_required(VERSION 3.4.1)用來設置在編譯本地庫時我們需要的最小的cmake版本,AndroidStudio自動生成,我們幾乎不需要自己管。

add_library( # Sets the nameof thelibrary.? ? ? ? ? ? native-lib? ? ? ? ? ? # Sets thelibrary as asharedlibrary.SHARED? ? ? ? ? ? # Provides a relative pathto your sourcefile(s).

src/main/cpp/native-lib.cpp )

add_library用來設置編譯生成的本地庫的名字為native-lib,SHARED表示編譯生成的是動態鏈接庫(這個概念前面已經提到過了),src/main/cpp/native-lib.cpp表示參與編譯的文件的路徑,這里面可以寫多個文件的路徑。

find_library是用來添加一些我們在編譯我們的本地庫的時候需要依賴的一些庫,由于cmake已經知道系統庫的路徑,所以我們這里只是指定使用log庫,然后給log庫起別名為log-lib便于我們后面引用,此處的log庫是我們后面調試時需要用來打log日志的庫,是NDK為我們提供的。

target_link_libraries是為了關聯我們自己的庫和一些第三方庫或者系統庫,這里把我們把自己的庫native-lib庫和log庫關聯起來。

NDK自定義配置

1 . 添加多個參與編譯的C/C++文件

首先,我們發現我們上面的例子都是涉及到一個C++文件,那么我們實際的項目不可能只有一個C++文件,所以我們首先要改變CMakeLists.txt文件,如下 :

add_library( HelloNDK? ? ? ? ? ? SHARED? ? ? ? ? ? src/main/cpp/HelloNDK.c? ? ? ? ? ? src/main/cpp/HelloJNI.c)

簡單吧,簡單明了,但是這里要注意的是,你在寫路徑的時候一定要注意當前的CMakeLists.txt在項目中的位置,上面的路徑是相對于CMakeLists.txt寫的。

2 . 我們想編譯出多個so庫

大家會發現,我們上面這樣寫,由于只有一個CMakeLists.txt文件,所以我們會把所有的C/C++文件編譯成一個so庫,這是很不合適的,這里我們就試著學學怎么編譯出多個so庫。

先放上我的項目文件夾結構圖:


然后看看我們每個CMakeLists.txt文件是怎么寫的:

one文件夾內的CMakeLists.txt文件的內容:

ADD_LIBRARY(one-lib SHAREDone-lib.c)target_link_libraries(one-liblog)

two文件夾內的CMakeLists.txt文件的內容:

ADD_LIBRARY(two-lib SHAREDtwo-lib.c)target_link_libraries(two-liblog)

app目錄下的CMakeLists.txt文件的內容

# Sets the minimum version of CMake required to build the native library.cmake_minimum_required(VERSION3.4.1)add_library( HelloNDK? ? ? ? ? ? SHARED? ? ? ? ? ? src/main/cpp/HelloNDK.c? ? ? ? ? ? src/main/cpp/HelloJNI.c)find_library(# Sets the name of the path variable.? ? ? ? ? ? ? log-lib# Specifies the name of the NDK library that# you want CMake to locate.? ? ? ? ? ? ? log )target_link_libraries(HelloNDK log)ADD_SUBDIRECTORY(src/main/cpp/one)ADD_SUBDIRECTORY(src/main/cpp/two)

通過以上的配置我們可以看出CMakeLists.txt文件的配置是支持繼承的,所以我們在子配置文件中只是寫了不同的特殊配置項的配置,最后在最上層的文件中配置子配置文件的路徑即可,現在編譯項目,我們會在<項目目錄>\app\build\intermediates\cmake\debug\obj\armeabi下面就可以看到生成的動態鏈接庫。而且是三個動態鏈接庫

3 . 更改動態鏈接庫生成的目錄

我們是不是發現上面的so庫的路徑太深了,不好找,沒事,可以配置,我們只需要在頂層的CMakeLists.txt文件中加入下面這句就可以了

#設置生成的so動態庫最后輸出的路徑set(CMAKE_LIBRARY_OUTPUT_DIRECTORY${PROJECT_SOURCE_DIR}/src/main/jniLibs/${ANDROID_ABI})

然后我們就可以在app/src/main下看到jniLibs目錄,在其中看到我們的動態鏈接庫的文件夾和文件(這里直接配置到了系統默認的路徑,如果配置到其他路徑需要在gradle文件中使用jinLibs.srcDirs = ['newDir']進行指定)。

NDK錯誤調試

在開發的過程中,難免會遇到bug,那怎么辦,打log啊,下面我們就談談打log和看log的姿勢。

1 . 在C/C++文件中打log

(1) 在C/C++文件中添加頭文件

#include

1

上面是打印日志的頭文件,必須添加

(2) 添加打印日志的宏定義和TAG

//log定義#defineLOG"JNILOG"http:// 這個是自定義的LOG的TAG#define? LOGD(...)? __android_log_print(ANDROID_LOG_DEBUG,LOG,__VA_ARGS__)// 定義LOGD類型#define? LOGI(...)? __android_log_print(ANDROID_LOG_INFO,LOG,__VA_ARGS__)// 定義LOGI類型#define? LOGW(...)? __android_log_print(ANDROID_LOG_WARN,LOG,__VA_ARGS__)// 定義LOGW類型#define LOGE(...)? __android_log_print(ANDROID_LOG_ERROR,LOG,__VA_ARGS__)// 定義LOGE類型#define LOGF(...)? __android_log_print(ANDROID_LOG_FATAL,LOG,__VA_ARGS__)// 定義LOGF類型

上面的日志級別和Android中的log是對應的。

(3) 經過上面兩步,我們就可以打印日志啦

intlen =5;LOGE("我是log %d",len);

現在我們就可以在logcat中看到我們打印的日志啦。

2 . 查看報錯信息

首先我們先手動寫一個錯誤,我們在上面的C文件中找一個函數,里面寫入如下代碼:

int * p = NULL;*p =100;

上面是一個空指針異常,我們運行程序,發現崩潰了,然后查看控制臺,只有下面一行信息:

libc:Fatal signal 11 (SIGSEGV), code 1, fault addr 0x0 in tid 17481

完全看不懂上面的信息好吧,這個也太不明顯了,下面我們就學習一下如何將上面的信息變得清楚明了

我們需要用到是ndk-stack工具,它在我們的ndk根目錄下,它可以幫助我們把上面的信息轉化為更為易懂更詳細的報錯信息,下面看看怎么做:

(1) 打開AndroidStudio中的命令行,輸入adb logcat > log.txt

上面這句我們是使用adb命令捕獲log日志并寫入log.txt文件,然后我們就可以在項目根目錄下看到log.txt文件

(2) 將log.txt打開看到報錯信息,如下:

F/libc? ? (17481): Fatal signal11 (SIGSEGV), code1, fault addr0x0 in tid17481 (dekong.ndkdemo1)I/DEBUG? (67): *** *** *** *** *** *** *** *** *** *** *** *** *** *** *** ***I/DEBUG? (67): Build fingerprint:'generic/vbox86p/vbox86p:5.0/LRX21M/genymotion08251046:userdebug/test-keys'I/DEBUG? (67): Revision:'0'I/DEBUG? (67): ABI:'x86'I/DEBUG? (67): pid:17481, tid:17481, name: dekong.ndkdemo1? >>> com.codekong.ndkdemo1 <<I/DEBUG? (67):? ? #03 pc a4016838? I/DEBUG? (67):I/DEBUG? (67): Tombstone written to: /data/tombstones/tombstone_05

現在的報錯信息還是看不懂,所以我們需要使用ndk-stack轉化一下:

(3) 繼續在AndroidStudio中的命令行中輸入如下命令(在這之前,我們必須要將ndk-stack的路徑添加到環境變量,以便于我們在命令行中直接使用它)

ndk-stack-sym app/build/intermediates/cmake/debug/obj/x86-dump./log.txt

上面的-sym后面的參數為你的對應平臺(我是Genymotion模擬器,x86平臺)的路徑,如果你按照上面的步驟改了路徑,那就需要寫改過的路徑,-dump后面的參數就是我們上一步得出的log.txt文件,執行結果如下:

********** Crash dump:**********Build fingerprint:'generic/vbox86p/vbox86p:5.0/LRX21M/genymotion08251046:userdebug/test-keys'pid:17481, tid:17481, name: dekong.ndkdemo1>>> com.codekong.ndkdemo1<<: Unableto open symbol file app/build/intermediates/cmake/debug/obj/x86/. Error (22): Invalid argumentStack frame I/DEBUG? (67):#03 pc a4016838: Unableto open symbol file app/build/intermediates/cmake/debug/obj/x86/. Error (22): Invalid argument

Crash dump is completed

尤其是上面的一句:

g_ndkdemo1_MainActivity_updateFile+150): Routine Java_com_codekong_ndkdemo1_MainActivity_updateFile at F:\AndroidFirstCode\NDKDemo1\app\src\main\cpp/HelloJNI.c:32

準確指出了發生錯誤的行數

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,505評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,556評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,463評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,009評論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,778評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,218評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,281評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,436評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,969評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,795評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,993評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,537評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,229評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,659評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,917評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,687評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,990評論 2 374

推薦閱讀更多精彩內容