1 // 用JNI實現 2 // 實例: 3 4 // 創建HelloWorld.java 5 class HelloWorld 6 { 7 private native void print(); 8 public static void main(String[] args) 9 { 10 new HelloWorld().print(); 11 } 12 13 static 14 { 15 System.loadLibrary("HelloWorld"); 16 } 17 } 18 // 注意print方法的聲明,關鍵字native表明該方法是一個原生代碼實現的。另外注意static代碼段的System.loadLibrary調用,這段代碼表示在程序加載的時候,自動加載libHelloWorld.so庫。 19 // 編譯HelloWorld.java 20 // 在命令行中運行如下命令: 21 javac HelloWorld.java 22 // 在當前文件夾編譯生成HelloWorld.class。 23 // 生成HelloWorld.h 24 // 在命令行中運行如下命令: 25 javah -jni HelloWorld 26 // 在當前文件夾中會生成HelloWorld.h。打開HelloWorld.h將會發現如下代碼: 27 /* DO NOT EDIT THIS FILE - it is machine generated */ 28 #include <jni.h> 29 /* Header for class HelloWorld */ 30 31 #ifndef _Included_HelloWorld 32 #define _Included_HelloWorld 33 #ifdef __cplusplus 34 extern "C" { 35 #endif 36 /* 37 * Class: HelloWorld 38 * Method: print 39 * Signature: ()V 40 */ 41 JNIEXPORT void JNICALL Java_HelloWorld_print 42 (JNIEnv *, jobject); 43 44 #ifdef __cplusplus 45 } 46 #endif 47 #endif 48 // 該文件中包含了一個函數Java_HelloWorld_print的聲明。這里面包含兩個參數,非常重要,后面講實現的時候會講到。 49 // 實現HelloWorld.c 50 // 創建HelloWorld.c文件輸入如下的代碼: 51 #include <jni.h> 52 #include <stdio.h> 53 #include "HelloWorld.h" 54 55 JNIEXPORT void JNICALL 56 Java_HelloWorld_print(JNIEnv *env, jobject obj) 57 { 58 printf("Hello World!\n"); 59 } 60 // 注意必須要包含jni.h頭文件,該文件中定義了JNI用到的各種類型,宏定義等。 61 // 另外需要注意Java_HelloWorld_print的兩個參數,本例比較簡單,不需要用到這兩個參數。但是這兩個參數在JNI中非常重要。 62 // env代表 java虛擬機環境,Java傳過來的參數和c有很大的不同,需要調用JVM提供的接口來轉換成C類型的,就是通過調用env方法來完成轉換的。 63 // obj代表調用的對象,相當于c++的this。當c函數需要改變調用對象成員變量時,可以通過操作這個對象來完成。 64 // 編譯生成libHelloWorld.so 65 // 在 Linux下執行如下命令來完成編譯工作: 66 cc -I/usr/lib/jvm/java-6-sun/include/linux/ 67 -I/usr/lib/jvm/java-6-sun/include/ 68 -fPIC -shared -o libHelloWorld.so HelloWorld.c 69 // 在當前目錄生成libHelloWorld.so。注意一定需要包含Java的include目錄(請根據自己系統環境設定),因為Helloworld.c中包含了jni.h。 70 // 另外一個值得注意的是在HelloWorld.java中我們LoadLibrary方法加載的是“HelloWorld”,可我們生成的Library卻是libHelloWorld。這是Linux的鏈接規定的,一個庫的必須要是:lib+庫名+.so。鏈接的時候只需要提供庫名就可以了。 71 // 運行Java程序HelloWorld 72 // 大功告成最后一步,驗證前面的成果的時刻到了: 73 java HelloWorld 74 // 如果你這步發生問題,如果這步你收到java.lang.UnsatisfiedLinkError異常,可以通過如下方式指明共享庫的路徑: 75 java -Djava.library.path='.' HelloWorld 76 // 當然還有其他的方式可以指明路徑請參考《在Linux平臺下使用JNI》。 77 // 我們可以看到久違的“Hello world!”輸出了。 |