首页 > 代码库 > java通过JNI调用c++代码

java通过JNI调用c++代码


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

}

}

注意命名,JniCpplibJniCpp.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

较详细的介绍JNI


http://blog.csdn.net/zjq2008wd/article/details/17582535

LinuxEclipse进行C++编程动态库so的生成与使用


http://www.blogjava.net/miaoyachun/archive/2012/12/06/392529.html

关于java.library.path


java通过JNI调用c++代码