您好,登錄后才能下訂單哦!
本篇文章為大家展示了如何通過JNI本地編寫來解決Java無法編寫大問題,內(nèi)容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲。
在您的程序中使用JNI并不是一件容易的事情,然而,JNI的性能和使用原有代碼的能力將會為您的Java程序添加更多的功能并且能勝任更多的挑戰(zhàn)。
Java本地接口(JNI——Java Native Interface)允許運行在Java虛擬機(jī)(JVM——Java Virtual Machine)上的代碼調(diào)用本地程序和類庫,或者被它們調(diào)用,這些程序和類庫可以是其它語言編寫的,比如C、C++或者匯編語言。當(dāng)一個程序無法完全使用Java編寫時,開發(fā)者可以通過JNI來編寫本地方法,比如標(biāo)準(zhǔn)Java類庫并不支持的依賴于平臺的特色或者程序庫。JNI還可以用于修改現(xiàn)有的使用其它語言編寫的程序,使它們可以通過Java編寫的程序來訪問。很多基本類庫都依賴JNI來為開發(fā)者和用戶提供服務(wù),比如文件的輸入/輸出和音頻功能。在基本類庫中包含的對于性能和平臺敏感的API可以允許所有的Java程序以安全和平臺無關(guān)的方式來使用這些功能,在采用JNI之前,開發(fā)者需要明確這些功能并不是已經(jīng)包含在Java標(biāo)準(zhǔn)類庫中的,在這篇文章中,我將會講解JNI是如何工作的以及本地類型是如何映射到Java的類型和類庫的。
JNI工作原理
在JNI中,本地函數(shù)是通過一個獨立的.c或.cpp文件來實現(xiàn)的(C++為JNI提供的界面會更簡潔一些)。當(dāng)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接口的結(jié)構(gòu),它包含了與JVM進(jìn)行交互以及與Java對象協(xié)同工作所必需的函數(shù),示例中的JNI函數(shù)可以在本地數(shù)組和Java數(shù)組類型之間、本地字符串和Java字符串類型之間進(jìn)行轉(zhuǎn)換,其功能還包括對象的實例化、拋出異常等?;旧夏梢允褂肑NIEnv來實現(xiàn)所有Java能做到的事情,雖然要簡單很多。更加正式的解釋是這樣的,本地代碼通過調(diào)用JNI的函數(shù)來訪問JVM,這是通過一個界面指針實現(xiàn)的(界面指針實際上是指向指針的指針),該指針指向一個指針數(shù)組,數(shù)組中的每個指針都指向了一個界面函數(shù),而每個界面函數(shù)都是在數(shù)組中預(yù)先定義過的。本地方法將JNI界面指針當(dāng)作一個參數(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)則遵循一個標(biāo)準(zhǔn)的本地化平臺的方式來轉(zhuǎn)換類庫的名稱到一個本地類庫的名稱。例如,在Solaris操作系統(tǒng)中會將pkg_Cls轉(zhuǎn)換為libpkg_Cls.so,而Win32系統(tǒng)則會將同樣的pkg_Cls轉(zhuǎn)換為pkg_Cls.dll。
動態(tài)指針會根據(jù)它們的名字來進(jìn)行解析,一個本地方法的名稱是按照組件進(jìn)行連接的,它包含了:前綴“Java_”、一個分離的合法的類名稱和一個分離的方法名稱。注意:微軟的JVM有相同的機(jī)制從Java調(diào)用本地Windows代碼,該機(jī)制被稱為原始本地接口(Raw Native Interface (RNI))?;绢愋停热缯?、字符等等,是在Java和本地代碼間進(jìn)行拷貝的,而其他的自定義Java對象則是通過引用來傳遞的。這個表格展示了Java和本地代碼之間的類型映射,這些類型是可以互換的,您可以在您使用int類型的位置使用jint類型,當(dāng)然反過來也一樣,而且不需要任何類型轉(zhuǎn)化。但是,Java的字符串和數(shù)組類型和本地的字符串與數(shù)組類型之間的轉(zhuǎn)換就比較困難了,如果您使用的jstring類型中出現(xiàn)了字符“*”,您的代碼會造成JVM的崩潰,以下的例子說明了您應(yīng)當(dāng)如何正確使用字符串:
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對象。
上述內(nèi)容就是如何通過JNI本地編寫來解決Java無法編寫大問題,你們學(xué)到知識或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識儲備,歡迎關(guān)注億速云行業(yè)資訊頻道。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。