介紹了在Android Studio
中配置NDK的開發環境:Android Studio NDK開發-環境配置,NDK開發環境配置完成以後,就要寫一下著名的Hello World
程序了。
在開始以前須要先介紹下Java和c/c++通訊:JNI,JNI(Java Native Inteface)是Java平臺的一部分,它容許Java代碼和其餘語言寫的代碼進行交互。尤爲是c/c++,可是並不妨礙你使用其餘語言,只要調用約定支持就能夠了。NDK上Java和c/c++有兩種交互方式:javascript
這裏使用的是第一種交互方式,即Java調用c/c++。java
##建立JNI目錄
打開Android Studio
,新建一個項目,右鍵點擊App
(對應的module)新建一個JNI目錄:android
jni
目錄:
##新建Java類
新建一個HelloJni.java的類,用來和NDK交互:c++
package com.example.jjz.jni;
public class HelloJni {
//定義一個jni的方法
public native String sayHello();
}複製代碼
HelloJni中使用關鍵詞native
定義了一個方法,native
:一個Natvie Method就是一個Java調用非Java代碼的接口,該方法的實現由c/c++實現。
方法定義完以後能夠看到一個提示,沒有sayHello
的實現。git
這是由於並無實現對應的c/c++的方法,下面就須要使用c/c++使用sayHello
方法。github
##gradle中支持NDKapp
Android Studio
從1.3開始就支持了對於NDK的開發,能夠在gradle配置NDK的開發選項,首先在bulid.gradle
中設置ndk的的moduleName:函數
defaultConfig {
applicationId "com.example.jjz.jni"
minSdkVersion 15
targetSdkVersion 23
versionCode 1
versionName "1.0"
//指定.so的目錄
sourceSets.main{
jniLibs.srcDir 'src/main/libs'
}
ndk{
moduleName 'hello'
}
}複製代碼
在進行同步gradle的時候出現了一個錯誤:post
NDK integration is deprecated in the current plugin
set "android.useDeprecatedNdk=true"in gradle.properties to continue using the current NDK integration複製代碼
根據提示須要在gradle.properties中設置android.useDeprecatedNdk=true
。gradle
#生成.h文件
Java
中使用調用NDK的方法,要先生成.h頭文件,JNI
的.h文件規則:
JNIEXPORT
.
變成_
。_
分割例如上面的sayHello
方法轉換爲JNI的方法爲:
JNIEXPORT jint JNICALL
Java_com_example_jjz_jni_HelloJni_sayHello(JNIEnv *env, jobject instance, jint a) {
}複製代碼
這種定義規則很複雜,不容易手寫,還好能夠經過javah
命令自動生成。
在app/目錄下運行命令:
javah -d src/main/jni/ -classpath build/intermediates/classes/debug/ com.example.jjz.jni.HelloJni
其中-d
是生成.h文件的保存目錄。-classpath
是指定.class所在的目錄,項目build成功以後,會在build/intermediates/classes/debug/
目錄裏生成.class文件。com.example.jjz.jni.HelloJni
是包名加上類名。
就能夠在jni目錄下獲得一個com_example_jjz_jni_HelloJni.h
的文件,文件內容以下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/
* Header for class com_example_jjz_jni_HelloJni */
#ifndef _Included_com_example_jjz_jni_HelloJni
#define _Included_com_example_jjz_jni_HelloJni
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_example_jjz_jni_HelloJni
* Method: sayHello
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_example_jjz_jni_HelloJni_sayHello
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif複製代碼
.h
文件只是一個聲明文件,還須要實現.h文件中定義的方法
添加.c文件
新建文件com_example_jjz_jni_HelloJni.c
用來文件實現sayHello
方法。
#include "com_example_jjz_jni_HelloJni.h"
JNIEXPORT void JNICALL JNIEXPORT jstring JNICALL Java_com_example_jjz_jni_HelloJni_sayHello(JNIEnv *env, jobject object) {
return (*env)->NewStringUTF(env, "Hello Jni");
}複製代碼
添加Application.mk文件
APP_MODULES := hello
APP_ABI :=all複製代碼
添加Android.mk文件
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE :=hello
LOCAL_SRC_FILES =: com_example_jjz_jni_HelloJni.c
include $(BUILD_SHARED_LIBRARY)複製代碼
##編譯調用
文件添加完成以後就可使用NDK編譯了,在../app/src/main/jni目錄下,運行命令
ndk-build
運行以後能夠看到生成的libhello.so文件:
注意這個時候.so
文件已經生成,可是在java中並無加載.so
類庫,必須加載.so
類庫才能被Java使用,加載的方式使用使用函數System.loadLiabrary("hello")
:
public class HelloJni {
static {
//加載.so類庫,加載的名稱去掉lib
System.loadLibrary("hello");
}
public native int sayHello(int a);
}複製代碼