jni编程最原始的:
eg:
dzt@dzt-laptop:~$ javac Abc.java
生成class文件
dzt@dzt-laptop:~$ javah Abc
根据class文件,生成头文件
编写abc.c文件
#include "Abc.h"
JNIEXPORT jint JNICALL Java_Abc_getAbc
(JNIEnv *env, jobject t)
{
return 0;
}
编译o文件:
gcc -fPIC -D_REENTRANT -I /usr/lib/jvm/jdk1.6.0_45/include -I /usr/lib/jvm/jdk1.6.0_45/include/linux/ -c abc.c
生成so文件:
gcc -shared abc.o -o libabc.so
把so库文件拷到 /usr/lib/jvm/jdk1.6.0_45/jre/lib/amd64/下 (我的机器是amd64,有的人是i386)
java Abc执行class文件
打印0;
==================使用jnative.jar包
如上面说的你如果要生成可以调用so库,你需要修改函数命,数据类型等,然后从新编译so库,相当的麻烦
使用jnative包不需要修改c代码,从新编译动态库文件,
使用步骤:
1,java项目增加jnative的引用
2,有的文档说JnativeCpp.so库要拷贝到/usr/lib/下,dll的话拷贝到system32下,
否则会出现错误:Jnative library not loaded.
java调用代码:
JNative jlib_msrwrite = new JNative("libic405.so", "test");
测试通过,拷贝在项目目录下也是可以的,使用相对目录如下:
JNative jlib_msrwrite = new JNative("./libic405.so", "test");
jlib_msrwrite.setRetVal(Type.INT);
jlib_msrwrite.setParameter(0, 1);//第一个参数
jlib_msrwrite.setParameter(1, 2);//第二个参数
jlib_msrwrite.setParameter(2, "12345");
jlib_msrwrite.setParameter(3, "1234");
jlib_msrwrite.setParameter(4, 10);
jlib_msrwrite.invoke();
String ret = jlib_msrwrite.getRetVal();
System.out.println("ret = " + ret);//输出函数返回结果
在c里字符指针char* 用 Pointer表示:
Pointer pstrack_Data2 = creatPointer(256);
Pointer pstrack_Data3 = creatPointer(256);
jlib_msrread.setRetVal(Type.INT);
jlib_msrread.setParameter(0, 1);
jlib_msrread.setParameter(1, 2);
jlib_msrread.setParameter(2, 23);
jlib_msrread.setParameter(3, pstrack_Data2);
jlib_msrread.setParameter(4, pstrack_Data3);
jlib_msrread.setParameter(5, 10);
jlib_msrread.invoke();
String ret = jlib_msrread.getRetVal();
String str1 = pstrack_Data2.getAsString();
String str2 = pstrack_Data3.getAsString();
System.out.println("str1:"+str1+"\nstr2:"+str2);//输出函数返回结果
其实java的代码很少很简单
对了,题目说是调用dll ,其实调用dll的java代码和调用so的是一样的,都是使用jnative.jar这个包。