轉載http://www.newhua.com/2008/0328/33542_2.shtml
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");
}
}