学会 jni

来源:互联网 发布:宝贝排名淘宝工具箱 编辑:程序博客网 时间:2024/05/01 09:30

       推荐视频:传智播客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

原创粉丝点击