步驟 2:編譯 Java 代碼
接下來,我們需要將 Java 代碼編譯成位元組碼。
完成這一步的方法之一是使用隨SDK一起提供的Java編譯器javac。
用來將 Java 代碼編譯成位元組碼的命令是:
cd test
javac JNI_javaCallc_test.java
如果是在eclipse環境下編寫的以上代碼,檔案儲存時會自動在工程目錄的bin下產生以上java檔案
步驟 3:建立 C/C++ 標頭檔
第三步是建立 C/C++ 標頭檔,它定義本機函數說明。
完成這一步的方法之一是使用 javah.exe,它是隨 SDK 一起提供的本機方法 C 存根產生器工具。
這個工具被設計成用來建立標頭檔,該標頭檔為在 Java 原始碼檔案中所找到的每個 native 方法定義 C 風格的函數。
這裡使用的命令是:
cd test
javah -classpath . test.JNI_javaCallc_test
注意.和test之間有空格
會產生以下檔案:
/* DO NOT EDIT THIS FILE - it is machine generated */#include <jni.h>/* Header for class test_JNI_javaCallc_test */#ifndef _Included_test_JNI_javaCallc_test#define _Included_test_JNI_javaCallc_test#ifdef __cplusplusextern "C" {#endif/* * Class: test_JNI_javaCallc_test * Method: intMethod * Signature: (I)I */JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intMethod (JNIEnv *, jobject, jint);/* * Class: test_JNI_javaCallc_test * Method: booleanMethod * Signature: (Z)Z */JNIEXPORT jboolean JNICALL Java_test_JNI_1javaCallc_1test_booleanMethod (JNIEnv *, jobject, jboolean);/* * Class: test_JNI_javaCallc_test * Method: stringMethod * Signature: (Ljava/lang/String;)Ljava/lang/String; */JNIEXPORT jstring JNICALL Java_test_JNI_1javaCallc_1test_stringMethod (JNIEnv *, jobject, jstring);/* * Class: test_JNI_javaCallc_test * Method: intArrayMethod * Signature: ([I)I */JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intArrayMethod (JNIEnv *, jobject, jintArray);#ifdef __cplusplus}#endif#endif
關於 C/C++ 標頭檔
如您可能已經注意到的那樣,JNI_javaCallc_test.h 中的 C/C++ 函數說明和 JNI_javaCallc_test.java 中的 Java native 方法聲明有很大差異。
1、JNIEXPORT 和 JNICALL 是用於匯出函數的、依賴於編譯器的指示符。
2、傳回型別、參數類型是映射到 Java 類型的 C/C++ 類型,比如:jstring,jint
現在來介紹下JNI裡的資料類型:
在C++裡,編譯器會很據所處的平台來為一些基本的資料類型來分配長度,因此也就造成了平台不一致性,而這個問題在Java中則不存在,因為有JVM的緣故,所以Java中的基礎資料型別 (Elementary Data Type)在所有平台下得到的都是相同的長度,比如int的寬度永遠都是32位。基於這方面的原因,java和c++的基礎資料型別 (Elementary Data Type)就需要實現一些mapping,保持一致性。
下面的表可以概括:下標列舉了常見的c/c++到到java的類型映射表。
| Java類型 |
本地類型 |
JNI中定義的別名 |
| int |
long |
jint |
| long |
_int64 |
jlong |
| byte |
signed char |
jbyte |
| boolean |
unsigned char |
jboolean |
| char |
unsigned short |
jchar |
| short |
short |
jshort |
| float |
float |
jfloat |
| double |
double |
jdouble |
| Object |
_jobject* |
jobject
|
JNI的設計者其實已經幫我們取好了相應的別名以方便記憶。如果想瞭解一些更加細緻的資訊,可以去看一些jni.h這個標頭檔,各種資料類型的定義以及別名就被定義在這個檔案中。
除了 Java 聲明中的一般參數以外,所有這些函數的參數表中都有一個指向 JNIEnv 和 jobject 的指標。
指向 JNIEnv 的指標實際上是一個指向函數指標表的指標。
正如將要在步驟4 中看到的,這些函數提供各種用來在C和C++中操作Java資料的能力。
jobject 參數引用當前對象
因此,如果C或C++代碼需要引用Java函數,則這個jobject 充當引用或指標,返回調用的 Java 對象。
函數名本身是由首碼“Java_”加全限定類名,再加底線和方法名構成的。