学会 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
- 学会 jni
- JAVA JNI 0基础学会简单编写程序
- 一个 Demo 学会搭建 Android Studio JNI 开发环境
- 学会
- 学会
- android加固系列—5.加固前先学会破解,hook(钩子)jni层系统api
- Android 一天搞定 NDK和JNI编程 3分钟学会写NDK实例
- Jni
- JNI
- JNI
- JNI
- JNI
- JNI
- JNI
- JNI
- JNI
- JNI
- JNI
- 2010湖南卫视“快乐男声”北美赛区海选正式开始
- c#窗体移动(无边框移动/用户区移动)
- WebGIS:一个图层的傻瓜式完整发布过程(一)
- 堆排序算法
- Repeater控件的ItemDataBound事件
- 学会 jni
- 抉择
- 流水帐
- IntelliJ 9 设置
- 在位图上写字
- 转载:基于Lua的自动化测试框架设计
- 今天开通CSDN博客,特此Mark一下,以后会经常更新!!!
- IntelliJ 9.0.2 注册码
- DLL(Dynamic Link Libraries)专题(转帖)