Java是通過JNI調用其他語言(包括C++)編譯的本地方法的,而本地方法是以庫文件的形式存放的(在WINDOWS平臺上是DLL文件形式,
在UNIX機器上是SO文件形式)。 如下是詳細講解:
1、JAVA中所需要做的工作 在JAVA程序中,首先需要在類中聲明所調用的庫名稱,如下: static {
System.loadLibrary(“goodluck”); } 在這裏,庫的擴展名字可以不用寫出來,究竟是DLL還是SO,由系統自己判斷。
還需要對將要調用的方法做本地聲明,關鍵字爲native。並且只需要聲明,而不需要具體實現。如下: public native static void
set(int i); public native static int get();
然後編譯該JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就會生成C/C++的頭文件。 例如程序TestDll.java,內容爲:
public class TestDll {
static {
System.loadLibrary("goodluck");
}
public native static int get();
public native static void set(int i);
public static void main(String[] args) {
TestDll test = new TestDll();
test.set(10);
System.out.println(test.get());
}
}
用javac TestDll.java編譯它,會生成TestDll.class。
再用javah -jni TestDll(最好是com.magic.test_jni.TesdDll這種形式,否則編譯可能有問題),則會在當前目錄下生成TestDll.h文件,這個文件需要被C/C++程序調用來生成所需的庫文件。
2、C/C++中所需要做的工作
對於已生成的.h頭文件,C/C++所需要做的,就是把它的各個方法具體的實現。然後編譯爲動態鏈接庫文件即可。再把庫文件拷貝到JAVA程序的路徑下面
,就可以用JAVA調用C/C++所實現的功能了。 接上例子。我們先看一下TestDll.h文件的內容:
// DO NOT EDIT THIS FILE
- it is machine generated
#include <jni.h> // Header for class TestDll
#ifndef _Included_TestDll
#define _Included_TestDll
#ifdef __cplusplus extern
"C" {
#endif
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass);
JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint);
#ifdef
__cplusplus
}
#endif
#endif
在具體實現的時候,我們只關心兩個函數原型
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass);
和 JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint);
這裏JNIEXPORT和JNICALL都是JNI的關鍵字,表示此函數是要被JNI調用的 。而jint是以JNI爲中介使JAVA的int類型與本地的int溝通的一種類型 ,我們可以視而不見,就當做int使用。函數的名稱是JAVA_再加上java程序的package路徑再加函數名組成的 。參數中,我們也只需要關心在JAVA程序中存在的參數,至於JNIEnv*和jclass我們一般沒有必要去碰它。
下面我們用TestDll.cpp文件具體實現這兩個函數:
#include "TestDll.h"
int i = 0;
JNIEXPORT jint JNICALL Java_TestDll_get (JNIEnv *, jclass) {
return i;
}
JNIEXPORT void JNICALL Java_TestDll_set (JNIEnv *, jclass, jint j) {
i = j;
}
編譯連接成庫文件,本例是在WINDOWS下做的,生成的是DLL文件。並且名稱要與JAVA中需要調用的一致,這裏就是goodluck.dll
把goodluck.dll拷貝到TestDll.class的目錄下,java TestDll運行它,就可以觀察到結果了。