Android Studio ndk-Jni開發詳解

兔子吃過窩邊草發表於2016-01-05

Java Native Interface (JNI)標準是java平臺的一部分,它允許Java程式碼和其他語言寫的程式碼進行互動。JNI 是本地程式設計介面,它使得在 Java 虛擬機器 (VM) 內部執行的 Java 程式碼能夠與用其它程式語言(如 C、C++ 和組合語言)編寫的應用程式和庫進行互動操作。

由於Android的應用層的類都是以Java寫的,這些Java類編譯為Dex型式的Bytecode之後,必須靠Dalvik虛擬機器(VM: Virtual Machine)來執行。VM在Android平臺裡,扮演很重要的角色。(多的我們不說了,介紹什麼的 到處都有可以去搜一下)

我們們先建一個NdkJniDemo的工程

Android Studio ndk-Jni開發詳細

新建JniUtils類實現native方法

public class JniUtils {    public static native String getStringFormC();
}

Android Studio ndk-Jni開發詳細

然後clean project 再rebuild project 生成class檔案,這時候開啟如下圖的資料夾看是否生成了classes資料夾,沒有生成請重新來過。

Android Studio ndk-Jni開發詳細

再開啟Terminal輸入指令cd app/build/intermediates/classes/debug

然後再輸入指令javah -jni com.wobiancao.ndkjnidemo.ndk.JniUtils

注意 這裡javah -jni後面跟的是JniUtils類的全路徑,如果javah報不存在之類的,是你的java環境沒有配置好。

Android Studio ndk-Jni開發詳細

這時候開啟classes/debug下面的檔案發現多了一個檔案com_wobiancao_ndkjnidemo_ndk_JniUtils.h

然後在src/main下新建資料夾jni,把生成的.h檔案複製或者剪下到jni資料夾下面去,新建一個c類隨便取一個名字,新增程式碼如下

//// Created by XY on 16/1/4.//#include "com_wobiancao_ndkjnidemo_ndk_JniUtils.h"/* 
* Class:     Java_com_wobiancao_ndkjnidemo_ndk_JniUtils 
* Method:    getStringFormC 
* Signature: ()Ljava/lang/String; 
*/JNIEXPORT jstring JNICALL Java_com_wobiancao_ndkjnidemo_ndk_JniUtils_getStringFormC
        (JNIEnv *env, jobject obj){ 
   return (*env)->NewStringUTF(env,"這裡是來自c的string");

這裡發現標頭檔案#include <jni.h>報紅色,是因為我們們還沒有配置ndk環境,開啟file->project structure
選擇你所下載的ndk環境路徑,如果沒有ndk這裡有個地址大家可以去下載 一個安卓工具集合的網站:http://androiddevtools.cn/

Android Studio ndk-Jni開發詳細

設定好了之後,發現標頭檔案還是紅色的,然後再build一下工程,就會有提示

Error: NDK integration is deprecated in the current plugin. Consider trying the new experimental plugin. For details, see http://tools.android.com/tech-docs/new-build-system/gradle-experimental. Set “android.useDeprecatedNdk=true” in gradle.properties to continue using the current NDK integration.

按著提示做就行了 在gradle.properties檔案末尾新增android.useDeprecatedNdk=true就ok啦

然後在app檔案下得build.gradle ->defaultConfig括號內新增如下程式碼

ndk {    
   moduleName "NdkJniDemo"          //生成的so名字 
   abiFilters "armeabi", "armeabi-v7a", "x86" //輸出指定三種abi體系結構下的so庫,目前可有可無。}

到了這一步重新build專案,發現已經沒有變紅了。接下來就是運用了,在JniUtils類裡面新增如下程式碼

static {    
   System.loadLibrary("NdkJniDemo");//之前在build.gradle裡面設定的so名字,必須一致}

然後簡單呼叫就行了,MainActivity程式碼如下

public class MainActivity extends AppCompatActivity { 
   TextView textView;    
@Override    protected void onCreate(Bundle savedInstanceState) { 
       super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView = (TextView) findViewById(R.id.ndk_text);
        textView.setText(JniUtils.getStringFormC());
    }
}

執行結果如圖

Android Studio ndk-Jni開發詳細

我們們開啟app->intermediates-ndk-debug發現生成了三個資料夾,並且對應了之前我們在build.gradle配置的abiFilters
大功告成。

新建libs資料夾把這三個資料夾放進去

Android Studio ndk-Jni開發詳細

然後刪除我們們的jni檔案試試,是否工程還能運用?

ok 教程完畢

最後出一個利用jni的Aes加密demo執行結果如圖所示

Android Studio ndk-Jni開發詳細

專案地址:https://github.com/a12a15a05/NdkJniDemo

相關文章