使用 Android Studio 寫出第一個 NDK 程式(超詳細)
NDK(Native Development Kit),是用於在 Android 應用中巢狀原生代碼的工具集,現在 Android 開發最常用的工具就是 Android Studio 了,筆者也是剛剛接觸 NDK 開發,用了一天的時間,踩了很多坑,最終跑通了自己的第一個 NDK 程式,話不多說,現在開始。
- 第一步:給 Android Studio 配置 NDK:
選擇 SDK Manager
選擇 SDK Tools
勾選 NDK 和一個 LLDB 版本,然後點選 Apply
接下來等待下載安裝即可,下載完成後,開啟 File -- Project Structure -- SDK Location
然後選擇 Android NDK location,點選 Select default NDK
然後點選 OK,就配置好了,然後開始寫程式了
從第二步開始就是相當於屬於 JNI(Java Native Interface | Java 本地介面) 開發的基礎流程,因為 NDK 是基於 JNI 的。
- 第二步:建立一個 JNITEST.java 的一個類:
public class JNITest {
//建立一個 native 方法
public native static String get();
}
注意:如果你建立的 get() 方法是紅色的,並且有這樣的提示:Cannot resolve corresponding JNI function Java_com_example_akon_jnitest_JNITest_get
解決辦法:File -- Settings -- Plugins
搜尋 NDK,然後將 Android NDK Support 後面的勾去掉,點選 OK,然後 Restart
當然,如果你沒有這種情況就不用管。
- 第三步:建立 C 語言檔案,建立 so 庫
點選 Make Project,生成 JNITest.class 檔案
使用 Project 方式檢視當前專案,在當前目錄下可以看到你的 JNITest.class 檔案
開啟 Android Studio 的 Terminal 到 移動到 app/src/main 目錄下
使用 javah -d jni -classpath F:\workSpace\Android\JNITest\app\build\intermediates\classes\debug com.example.akon.jnitest.JNITest 命令建立 .h 的標頭檔案
javah:生成標頭檔案
-d jni:當前目錄下建立一個 jni 資料夾
-classpath .../debug 指定要生成標頭檔案的位元組碼檔案目錄,即我們剛剛的 JNITest.class 的目錄
com.example.akon.jnitest.JNITest 是 JNITest 檔案的包名加上位元組碼檔案的名稱
注意:這個 debug 檔案目錄可能太長,輸入麻煩,我們可以找到 debug 資料夾,右鍵 copy path,複製檔案目錄即可
現在我們可以看到 app/src/main 目錄下有一個 jni 資料夾,裡面有一個 com_example_akon_jnitest_JNITest.h 的標頭檔案,就是我們生成的標頭檔案,標頭檔案命名也是按照包名加位元組碼名的規範,以下劃線連線。
然後在 jni 目錄下建立一個 c/c++ resource 檔案 test.c,要選擇 c 為字尾:
#include<jni.h>
#include<stdio.h>
//匯入我們建立的標頭檔案
#include "com_example_akon_jnitest_JNITest.h"
JNIEXPORT jstring JNICALL Java_com_example_akon_jnitest_JNITest_get
(JNIEnv *env, jclass jclass){
//返回一個字串
return (*env)->NewStringUTF(env,"This is my first NDK Application");
}
下面的這個方法使我們從標頭檔案中複製過來的然後修改了引數,給了返回值。
然後我們在 jni 目錄下建立兩個 .mk 檔案:
一個 Android.mk:
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := JNITest
LOCAL_SRC_FILES := test.c
include $(BUILD_SHARED_LIBRARY)
一個是 Application.mk:
APP_ABI := all
然後我們需要生成 so 庫:
開啟 Terminal,到 app/src/main/jni 目錄下,使用 ndk-build 命令生成 so 庫:
接著開啟 app/src/main/libs 就可以看見我們生成的 so 庫了。
為了防止 so 庫相容錯誤,在 gradle.properties 最後一行新增:
android.useDeprecatedNdk=true
為了讓專案能夠找到我們的 so 庫,在 build.gradle 資料夾的 android 下新增:
sourceSets {
main() {
jniLibs.srcDirs = ['src/main/libs']
jni.srcDirs = [] //遮蔽掉預設的jni編譯生成過程
}
}
然後我們在 JNITest.java 中動態匯入 so 庫,不需要寫 libJNITest,只用寫 JNITest:
package com.example.akon.jnitest;
public class JNITest {
// 動態匯入 so 庫
static {
System.loadLibrary("JNITest");
}
//建立一個 native 方法
public native static String get();
}
然後我們在 MainActivity 中列印 JNITest 的 get() 方法獲取到的 String 值:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//列印資訊
Log.e("Message",JNITest.get());
}
}
然後執行,檢視日誌,搜尋 Message:
然後我們就檢視到了我們獲取的 String 了,“This is my first NDK Application”,就是我們在 C 語言檔案中所返回的字串。
好了,第一個 NDK 程式就寫完了。