Java本地接口(Java Native Interface (JNI))允許運行在Java虛擬機(Java Virtual Machine (JVM))上的代碼調用本地程序和類庫,或者被它們調用,這些程序和類庫可以是其它語言編寫的,比如C、C++或者匯編語言。
當一個程序無法完全使用Java編寫時,開發者可以通過JNI來編寫本地方法,比如標准Java類庫並不支持的依賴於平台的特色或者程序庫。JNI還可以用於修改現有的使用其它語言編寫的程序,使它們可以通過Java編寫的程序來訪問。
很多基本類庫都依賴JNI來為開發者和用戶提供服務,比如文件的輸入/輸出和音頻功能。在基本類庫中包含的對於性能和平台敏感的API可以允許所有的Java程序以安全和平台無關的方式來使用這些功能,在采用JNI之前,開發者需要明確這些功能並不是已經包含在Java標准類庫中的,在這篇文章中,我將會講解JNI是如何工作的以及本地類型是如何映射到Java的類型和類庫的。
JNI工作原理
在JNI中,本地函數是通過一個獨立的.c或.cpp文件來實現的(C++為JNI提供的界面會更簡潔一些)。當JVM調用該函數時,它傳遞了一個JNIEnv指針、一個jobject指針和通過Java方法定義的Java參數,JNI函數的形式如下:
JNIEXPORT void JNICALL Java_ClassName_MethodName
(JNIEnv *env, jobjectobj)
{
//Method native implemenation
}
env指針是一個包含了JVM接口的結構,它包含了與JVM進行交互以及與Java對象協同工作所必需的函數,示例中的JNI函數可以在本地數組和Java數組類型之間、本地字符串和Java字符串類型之間進行轉換,其功能還包括對象的實例化、拋出異常等。基本上您可以使用JNIEnv來實現所有Java能做到的事情,雖然要簡單很多。
更加正式的解釋是這樣的,本地代碼通過調用JNI的函數來訪問JVM,這是通過一個界面指針實現的(界面指針實際上是指向指針的指針),該指針指向一個指針數組,數組中的每個指針都指向了一個界面函數,而每個界面函數都是在數組中預先定義過的。
本地方法將JNI界面指針當作一個參數,如果在同一個Java線程中,出現對該本地方法的多重調用,JVM則保證傳遞相同的界面指針到本地方法。不過,一個本地方法可以被不同的Java線程調用,因而也可能會收到不同的JNI界面指針。
本地方法是通過System.loadLibrary方法加載的,在以下的例子中,類的初始化方法加載了一個指定平台的本地類庫,該類庫定義了本地方法:
packagepkg;
class Cls {
native double f(inti, String s);
static {
System.loadLibrary(pkg_Cls");
}
}
System.loadLibrary方法的參數是一個類庫的名稱,它可以由程序員任意選取,系統則遵循一個標准的本地化平台的方式來轉換類庫的名稱到一個本地類庫的名稱。例如,在Solaris操作系統中會將pkg_Cls轉換為libpkg_Cls.so,而Win32系統則會將同樣的pkg_Cls轉換為pkg_Cls.dll。
動態指針會根據它們的名字來進行解析,一個本地方法的名稱是按照組件進行連接的,它包含了:前綴“Java_”、一個分離的合法的類名稱和一個分離的方法名稱。
注意:微軟的JVM有相同的機制從Java調用本地Windows代碼,該機制被稱為原始本地接口(Raw Native Interface (RNI))。
數據類型映射
基本類型,比如整型、字符等等,是在Java和本地代碼間進行拷貝的,而其他的自定義Java對象則是通過引用來傳遞的。
這個表格展示了Java和本地代碼之間的類型映射,這些類型是可以互換的,您可以在您使用int類型的位置使用jint類型,當然反過來也一樣,而且不需要任何類型轉化。但是,Java的字符串和數組類型和本地的字符串與數組類型之間的轉換就比較困難了,如果您使用的jstring類型中出現了字符“*”,您的代碼會造成JVM的崩潰,以下的例子說明了您應當如何正確使用字符串:
JNIEXPORT void JNICALL Java_ClassName_MethodName
(JNIEnv *env, jobjectobj, jstringjavaString)
{
//Get the native string from Java string
const char *nativeString = env->GetStringUTFChars(env,javaString, 0);
printf("%s", nativeString);
env->ReleaseStringUTFChars(env,javaString, nativeString);
}
您需要使用界面指針env來操作Java對象。
總結
在您的程序中使用JNI並不是一件容易的事情,然而,JNI的性能和使用原有代碼的能力將會為您的Java程序添加更多的功能並且能勝任更多的挑戰,如果需要關於JNI的更多信息,可以訪問JNI的主頁。