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 <iostream> #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>改为#include"jni.h"
编译出错:../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 -MF"hello.d"-MT"hello.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"); } }
参考资料:
http://blog.csdn.net/skywalker256/article/details/4677644
http://blog.csdn.net/zjq2008wd/article/details/17582535
Linux下Eclipse进行C++编程动态库so的生成与使用
http://www.blogjava.net/miaoyachun/archive/2012/12/06/392529.html