java通過JNI調用c++代碼
1 用java約定接口,生成頭文件
創建JniHelloWorld.java
public class JniHelloWorld { public JniHelloWorld(){ } public native void sayHello(String name); }
生成頭文件供c++使用
javah-jni JniHelloWorld
產生JniHelloWorld.h文件,裡面是用c++代碼規定了接口形式。
2生成動態鏈接庫文件(http://blog.csdn.net/zjq2008wd/article/details/17582535)
用eclipsec++這個IDE生成,new->project->c++project選擇SharedLibrary->Empty Project。
這裡命名為JniCpp。
創建一個cpp文件來實現頭文件的接口,命名為hello.cpp。
#include#include JniHelloWorld.h using namespace std; JNIEXPORT void JNICALL Java_JniHelloWorld_sayHello (JNIEnv* env, jobject obj, jstring name) { const char* pname = env->GetStringUTFChars(name, NULL); cout << Hello, << pname << endl; }
編譯出錯:../JniHelloWorld.h:2:17:fatal error: jni.h: No such file or directory
從/usr/local/lib/jdk1.7.0_55/include找到該文件,拷貝到工程裡面。
將JniHelloWorld.h的#include
編譯出錯:../jni.h:45:20:fatal error: jni_md.h: No such file or directory
同理,將/usr/local/lib/jdk1.7.0_55/include/linux包含到工程裡面。
鏈接出錯:/usr/bin/ld:./hello.o: relocation R_X86_64_32 against `.rodata' can not be usedwhen making a shared object; recompile with -fPIC
解決方法:project->properties->setting ->tool setting->complier如下圖所示添加-fPIC
把g++改為g++-fPIC。
(這裡我試過直接修改makefile,發現失敗,因為elicpse每次都會按照她的規則生成新的makefile)
成功了:
makeall
Buildingfile: ../hello.cpp
Invoking:GCC C++ Compiler
g++-fPIC -O0 -g3 -Wall -c -fmessage-length=0 -MMD -MP -MFhello.d-MThello.d -o hello.o ../hello.cpp
Finishedbuilding: ../hello.cpp
Buildingtarget: libJniCpp.so
Invoking:GCC C++ Linker
g++-shared -o libJniCpp.so ./hello.o
Finishedbuilding target: libJniCpp.so
這個libJniCpp.so就是我們所需要的動態鏈接庫。
3java調用動態鏈接庫
首先,讓java識別到這個so的位置。(http://www.blogjava.net/miaoyachun/archive/2012/12/06/392529.html)
我是直接在JniHelloWorld類裡面添加如下代碼:
static{
System.out.println(load);
try{
System.setProperty(java.library.path,
System.getProperty(java.library.path)+
:/home/linger/jni);
System.out.println(System.getProperty(java.library.path));
Field fieldSysPath =ClassLoader.class.getDeclaredField(sys_paths);
fieldSysPath.setAccessible(true);
fieldSysPath.set(null,null);
System.loadLibrary(JniCpp);
}catch(Exception e) {
// do nothing for exception
}
}
注意命名,JniCpp和libJniCpp.so,如果命名錯了也是會找不到動態鏈接庫的。
然後在這個類裡面添加main函數來測試,
public static voidmain(String[] args)
{
System.out.println(main);
JniHelloWorldshp =newJniHelloWorld();
shp.sayHello(World);
}
最後,JniHelloWorld.java文件是這樣子的:
import java.lang.reflect.Field; public class JniHelloWorld { static{ System.out.println(load); try { System.setProperty(java.library.path, System.getProperty(java.library.path)+ :/home/linger/jni); System.out.println(System.getProperty(java.library.path)); Field fieldSysPath = ClassLoader.class.getDeclaredField(sys_paths); fieldSysPath.setAccessible(true); fieldSysPath.set(null, null); System.loadLibrary(JniCpp); } catch (Exception e) { // do nothing for exception } } public JniHelloWorld(){ } public native void sayHello(String name); public static void main(String[] args) { System.out.println(main); JniHelloWorld shp = new JniHelloWorld(); shp.sayHello(World); } }