Java通過JNI調(diào)用C語言的方法
Java通過JNI調(diào)用本地方法,而本地方法是以庫文件的形式存放的(在WINDOWS平臺上是DLL文件形式,在UNIX機(jī)器上是SO文件形式)。通過調(diào)用本地的庫文件的內(nèi)部方法,使Java可以實(shí)現(xiàn)和本地機(jī)器的緊密聯(lián)系,調(diào)用系統(tǒng)級的各接口方法。
簡單介紹及應(yīng)用如下:
一、Java中所需要做的工作
在Java程序中,首先需要在類中聲明所調(diào)用的庫名稱,如下:
Java代碼
- static {
- System.loadLibrary(“goodluck”);
- }
- static {
- System.loadLibrary(“goodluck”);
- }
在這里,庫的擴(kuò)展名字可以不用寫出來,究竟是DLL還是SO,由系統(tǒng)自己判斷。
還需要對將要調(diào)用的方法做本地聲明,關(guān)鍵字為native。并且只需要聲明,而不需要具體實(shí)現(xiàn)。如下:
Java代碼
- public native static void set(int i);
- public native static int get();
- public native static void set(int i);
- public native static int get();
然后編譯該Java程序文件,生成CLASS,再用JavaH命令,JNI就會生成C/C++的頭文件。
例如程序testdll.Java,內(nèi)容為:
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());
- }
- }
- 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 testdll,則會在當(dāng)前目錄下生成testdll.h文件,這個(gè)文件需要被C/C++程序調(diào)用來生成所需的庫文件。
二、C/C++中所需要做的工作
對于已生成的.h頭文件,C/C++所需要做的,就是把它的各個(gè)方法具體的實(shí)現(xiàn)。然后編譯連接成庫文件即可。再把庫文件拷貝到Java程序的路徑下面,就可以用Java調(diào)用C/C++所實(shí)現(xiàn)的功能了。
接上例子。我們先看一下testdll.h文件的內(nèi)容:
Java代碼
- /* DO NOT EDIT THIS FILE - it is machine generated */
- #include
- /* Header for class testdll */
- #ifndef _Included_testdll
- #define _Included_testdll
- #ifdef __cplusplus
- extern "C" {
- #endif
- /*
- * Class: testdll
- * Method: get
- * Signature: ()I
- */
- JNIEXPORT jint JNICALL Java_testdll_get
- (JNIEnv *, jclass);
- /*
- * Class: testdll
- * Method: set
- * Signature: (I)V
- */
- JNIEXPORT void JNICALL Java_testdll_set
- (JNIEnv *, jclass, jint);
- #ifdef __cplusplus
- }
- #endif
- #endif
- /* DO NOT EDIT THIS FILE - it is machine generated */
- #include
- /* Header for class testdll */
- #ifndef _Included_testdll
- #define _Included_testdll
- #ifdef __cplusplus
- extern "C" {
- #endif
- /*
- * Class: testdll
- * Method: get
- * Signature: ()I
- */
- JNIEXPORT jint JNICALL Java_testdll_get
- (JNIEnv *, jclass);
- /*
- * Class: testdll
- * Method: set
- * Signature: (I)V
- */
- JNIEXPORT void JNICALL Java_testdll_set
- (JNIEnv *, jclass, jint);
- #ifdef __cplusplus
- }
- #endif
- #endif
在具體實(shí)現(xiàn)的時(shí)候,我們只關(guān)心兩個(gè)函數(shù)原型 :
Java代碼
JNIEXPORT jint JNICALL Java_testdll_get (JNIEnv *, jclass);
JNIEXPORT jint JNICALL Java_testdll_get (JNIEnv *, jclass);
和Java代碼
JNIEXPORT void JNICALL Java_testdll_set (JNIEnv *, jclass, jint);
JNIEXPORT void JNICALL Java_testdll_set (JNIEnv *, jclass, jint);
這里JNIEXPORT和JNICALL都是JNI的關(guān)鍵字,表示此函數(shù)是要被JNI調(diào)用的。而jint是以JNI為中介使Java的int類型與本地的int溝通的一種類型,我們可以視而不見,就當(dāng)做int使用。函數(shù)的名稱是Java_再加上Java程序的package路徑再加函數(shù)名組成的。參數(shù)中,我們也只需要關(guān)心在Java程序中存在的參數(shù),至于JNIEnv*和jclass我們一般沒有必要去碰它。
好,下面我們用testdll.cpp文件具體實(shí)現(xiàn)這兩個(gè)函數(shù):
Java代碼
- #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;
- }
- #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中需要調(diào)用的一致,這里就是goodluck.dll
把goodluck.dll拷貝到testdll.class的目錄下,Java testdll運(yùn)行它,就可以觀察到結(jié)果了。
【編輯推薦】