Java通過
JNI機(jī)制調(diào)用c/c++寫的native程序。c/c++開發(fā)的native程序需要遵循一定的JNI規(guī)范,下面的例子就是一個JNI函數(shù)聲明:
JNIEXPORT jint JNICALL Java_jnitest_MyTest_test
(JNIEnv * env, jobject obj, jint arg0);
JVM負(fù)責(zé)從Java Stack轉(zhuǎn)入C/C++ Native Stack。當(dāng)Java進(jìn)入JNI調(diào)用,除了函數(shù)本身的參數(shù)(arg0),會多出兩個參數(shù):JNIEnv指針和jobject指針。
JNIEnv指針是JVM創(chuàng)建的,用于Native的c/c++方法操縱Java執(zhí)行棧中的數(shù)據(jù),比如Java Class, Java Method等。
首先,JNI對于JNIEnv的使用, 提供了兩種語法: c語法以及c++語法,如下:
c語法:
jsize len = (*env)->GetArrayLength(env,array);
c++語法:
jsize len =env->GetArrayLength(array);
(注:由于C語言并不支持對象的概念,所以C語法中需要把env作為第一個參數(shù)傳入,類似于C++的隱式參數(shù)this指針).
另外: JNIEnv有幾個設(shè)計的原則:
第一、JNIEnv指針被設(shè)計成了
Thread Local Storage(TLS)變量,也就是說每一個Thread, JNIEnv變量都有獨(dú)立的Copy。這樣做的原因主要是考慮到:
由于JVM要運(yùn)行在多個平臺(除了主流的Windows,Linux等平臺),JNI
內(nèi)部實(shí)現(xiàn)很多要依賴到TLS, 為了減少對TLS的依賴,所有TLS based的數(shù)據(jù)都會存放于JNIEnv中。這樣相當(dāng)于只依賴一個TLS based的變量JNIEnv。由于JNIEnv指針是TLS的,所以你不能把Thead#1使用的JNIEnv傳給Thread#2使用。
第二、JNIEnv中定義了一組函數(shù)指針,c/c++ Native程序是通過這些函數(shù)指針操縱Java數(shù)據(jù)。這樣設(shè)計的好處是:你的c/c++ 程序不需要依賴任何函數(shù)庫,或者DLL。由于JVM可能由不同的廠商實(shí)現(xiàn),不同廠商有自己不同的JNI實(shí)現(xiàn),如果要求這些廠商暴露約定好的一些頭文件和庫,這不是靈活的設(shè)計。
而且使用函數(shù)指針表的另外一個好處是: JVM可以根據(jù)啟動參數(shù)動態(tài)替換JNI實(shí)現(xiàn)。比如:類似于C庫,JNI實(shí)現(xiàn)為了性能起見,并沒有對調(diào)用者傳入的參數(shù)進(jìn)行檢查。但是在調(diào)試階段,也許這種檢查是很必要的,幫助你盡早發(fā)現(xiàn)BUG。例如如果你使用IBM JDK,你可以指定JVM參數(shù)–Xcheck:jni,告訴JVM使用帶檢查的JNI實(shí)現(xiàn)。
參考:
http://java.sun.com/docs/books/jni/html/jniTOC.html