?
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
改为
#includejni.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 -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);
}
}
?
?