通過JNI調用C程式實現"Hello World!"輸出,掌握此方法可以很方便地將Java函數用C/C++實現。
確定安裝成功Java,本機上因為工程需要,已經安裝幾個jdk版本,執行java -version
root@wang:~/jni# java -version
openjdk version "1.8.0_91"
OpenJDK Runtime Environment (build 1.8.0_91-8u91-b14-0ubuntu4~14.04-b14)
OpenJDK 64-Bit Server VM (build 25.91-b14, mixed mode)
可以看到使用OpenJDK 8的版本。
編寫可以HelloWorld.java程式,有一個native方法聲明為DisplayHello.
public class HelloWorld {static {System.loadLibrary("Hello");}public native void DisplayHello();public static void main(String[] args) {new HelloWorld().DisplayHello();}}執行javac HelloWorld.java產生HelloWorld.class檔案
執行javah -jni HelloWorld會在目前的目錄下產生HelloWorld.h檔案
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class HelloWorld */ #ifndef _Included_HelloWorld #define _Included_HelloWorld #ifdef __cplusplus extern "C" { #endif /* * Class: HelloWorld * Method: DisplayHello * Signature: ()V */ JNIEXPORT void JNICALL Java_HelloWorld_DisplayHello (JNIEnv *, jobject); #ifdef __cplusplus } #endif #endif此檔案為javah自動產生的,不需要修改,可以看到裡麵包含一個Java_HelloWorld_DisplayHello的函式宣告。
編寫imp_hello.c程式,此程式實現剛剛聲明的函數。
#include <jni.h> #include "HelloWorld.h" #include <stdio.h> JNIEXPORT void JNICALL Java_HelloWorld_DisplayHello(JNIEnv * env, jobject obj) { printf("From imp_hello.c:"); printf(" Hello world ! \n"); return; }此檔案包括兩個標頭檔jni.h和產生的HelloWorld.h,實現函數中就列印一句話From imp_hello.c: Hello world !
產生動態庫命令:
gcc -shared -fPIC -I /usr/lib/jvm/java-8-openjdk-amd64/include/ -I /usr/lib/jvm/java-8-openjdk-amd64/include/linux/ imp_hello.c -o libHello.so
不同的機器可能jdk的路徑不相同,可以通過上面的命令查看使用的jdk版本,包含兩個檔案路徑,一個是jni.h檔案路徑,一個是jni_md.h的檔案路徑。如果碰到其它的錯誤,可以根據提示解決,比如本機上如果不包含-fPIC,就會提示出錯
root@wang:~/jni# gcc -shared -I /usr/lib/jvm/java-8-openjdk-amd64/include/ -I /usr/lib/jvm/java-8-openjdk-amd64/include/linux/ imp_hello.c -o libHello.so
/usr/bin/ld: /tmp/ccXqfTfA.o: relocation R_X86_64_32 against `.rodata' can not be used when making a shared object; recompile with -fPIC
/tmp/ccXqfTfA.o: error adding symbols: Bad value
collect2: error: ld returned 1 exit status
提示中已經給出解決方案,recompile with -fPIC,如果碰到其它的錯誤也可以上網查詢相關的問題。
編譯完成之後目前的目錄下產生libHello.so庫
root@wang:~/jni# ls
HelloWorld.class HelloWorld.h HelloWorld.java imp_hello.c libHello.so
最後執行
root@wang:~/jni# java -Djava.library.path=. HelloWorld
From imp_hello.c: Hello world !
此時Java程式已經成功調用C程式,如果是C++程式則可以將imp_hello.c命名為imp_hello.cpp,將gcc改成g++,其它步驟和此相同。