推荐视频:传智播客http://bt.itcast.cn/jniVideo.htm
前几天老师要求我写一个jni程序,调用一个叫短信猫的设备。可是设备只提供了一个在c++下的接口,而我们的开发环境是Java。这可有点郁闷了。最后自己学习了一下jni.现在就把经验分享给大家:
从写Java程序test.java到生成test.h文件就不多说了:
1) 编写java程序:
这里以HelloWorld为例。
代码1:
class HelloWorld {
public native void displayHelloWorld();
static {
System.loadLibrary("hello");
}
public static void main(String[] args) {
new HelloWorld().displayHelloWorld();
}
}
声明native方法:如果你想将一个方法做为一个本地方法的话,那么你就必须声明改方法为native的,并且不能实现。其中方法的参数和返回值在后面讲述。
Load动态库:System.loadLibrary("hello");加载动态库(我们可以这样理解:我们的方法 displayHelloWorld()没有实现,但是我们在下面就直接使用了,所以必须在使用之前对它进行初始化)这里一般是以static块进行加载的。同时需要注意的是System.loadLibrary();的参数“hello”是动态库的名字。
main()方法
2) 编译没有什么好说的了
javac HelloWorld.java
3) 生成扩展名为h的头文件
javah HelloWorld
头文件的内容:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class HelloWorld */
#ifndef _Included_HelloWorld
#define _Included_HelloWorld
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: HelloWorld
* Method: displayHelloWorld
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_HelloWorld_displayHelloWorld
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
(这里我们可以这样理解:这个h文件相当于我们在java里面的接口,这里声明了一个 Java_HelloWorld_displayHelloWorld (JNIEnv *, jobject);方法,然后在我们的本地方法里面实现这个方法,也就是说我们在编写C/C++程序的时候所使用的方法名必须和这里的一致)。
4) 编写本地方法
打开Vc,新建->; 工程 ->; win32 Dynamic- Link Library .在向导中选择空工程.
随便一个新建源文件 ,实现和由javah命令生成的头文件里面声明的方法名相同的方法。
代码2:
1 #include <jni.h>
2 #include "HelloWorld.h"
3 #include <stdio.h>
4 JNIEXPORT void JNICALL Java_HelloWorld_displayHelloWorld(JNIEnv *env, jobject obj)
{
printf("Hello world!/n");
return;
}
注意代码2中的第1行,需要将jni.h文件引入(假如不引入,会报错,提示在xx找不到jni.h ,jni在jdk/include/里边,复制到xx位置即可),因为在程序中的JNIEnv、 jobject等类型都是在该头文件中定义的;另外在第2行需要将HelloWorld.h头文件引入,最后编译即可。
5)在vc项目里找到.dll文件拷贝到第二步你生成的.class文件同目录中,用java HelloWorld命令运行即可
参考:http://java.chinaitlab.com/JDK/36677.html
需要注意的是
1、java和c是如何互通的。
其实不能互通的原因主要是数据类型的问题,jni解决了这个问题,例如那个c文件中的 jstring数据类型就是java传入的String对象,经过jni函数的转化就能成为c的char*。
对应数据类型关系如下表:
Java类型本地c类型说明boolean jboolean无符号 ,8位byte jbyte无符号,8位char jchar无符号,16位short jshort有符号,16位int jint有符号,32位long jlong有符号,64位float jfloat32位double jdouble64位void voidN/A。
2、如何将 java传入的String参数转换为c的char*,然后使用。
java传入的String 参数,在c文件中被jni转换为jstring的数据类型,在c文件中声明char* test,然后test=(char*)(*env)->;GetStringUTFChars(env,jstring,NULL);注意:test使用完后,通知虚拟机平台相关代码无需再访问: (*env)->;ReleaseStringUTFChars(env,jstring,test);
3、将c中获取的一个 char*的buffer传递给java。
这个char*如果是一般的字符串 的话,作为string传回去就可以了。如果是含有’/0’的buffer,最好作为byte array传出,因为可以制定copy的length,如果copy到string,可能到’/0’就截断了。
有两种方式传递得到的数据:
一种是在jni中直接new一个byte数组,然后调用函数 (*env)->;SetByteArrayRegion(env,bytearray,0,len,buffer);将buffer的值copy 到bytearray中,函数直接return bytearray就可以了。
一种是return错误号,数据作为参数传出,但是 java的基本数据类型是传值 ,对象是传递的引用,所以将这个需要传出的 byte数组用某个类包一下,如下:
class RetObj{public byte[]bytearray;}
这个对象作为函数的参数retobj传出,通过如下函数将retobj中的byte数组赋值便于传出。
代码如下:jclasscls;jfieldIDfid;jbyteArraybytearray;bytearray=(*env)-& gt;;NewByteArray(env,len); (*env)->;SetByteArrayRegion(env,bytearray,0,len,buffer);cls=(*env)-& gt;;GetObjectClass(env,retobj);fid=(*env)-& gt;;GetFieldID(env,cls,"retbytes","[B"]); (*env)->;SetObjectField(env,retobj,fid,bytearray);
参考:http://www.hudong.com/wiki/JNI