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