java与c/c++之间的数据交互-----jni点滴【转】
最近作一个tiemsten数据库的项目,用到了jni技术。在这个项目中,我们用java来写界面和业务逻辑,用c语言写数据库odbc访问。单纯的odbc其实没有什么难的,但是在java和c之间进行数据传递是比较麻烦的事情。两者之间数据的传递有这样几种情况:java和c之间基本数据类型的交互,java向c传递对象类型,c向java返回对象类型,c调用java类。下面就这样几种情况分类说明。
1、java 向c传递基本数据类型
对于基本数据类型,java和c是相互对应的,所以可以直接使用。它们的对应关系为;
------------------------------------------------------------------------
Java类型 本地类型 字节(bit)
-------------------------------------------------------------------------
boolean
jboolean 8, unsigned
byte jbyte 8
char jchar 16,
unsigned
short jshort 16
int jint
32
long jlong 64
float jfloat
32
double jdouble 64
void void n/a
------------------------------------------------------------------------
2.java向c传递对象类型
对于java传递进来的java对象模型,c要加载java类的原型,根据创建相应的c对象,获取java对象的方法的id,然后调用java对象的方法。举例说明:比如有个java类customer对象作为jni参数传递到c程序,customer有方法String
getName()。
JNIEXPORT jobject JNICALL
Java_com_oracle_estt_sc_db_impl_SCQueryODBC__1getCustomer
(JNIEnv *env,
jobject, jobject customer){
jmethodID methodId;
//获得customer对象的句柄
jclass
cls_objClass=env->GetObjectClass(customer);
//获得customer对象中特定方法getName的id
methodId=env->GetMethodID(cls_objClass,"getName","()Ljava/lang/String;");
//调用customer对象的特定方法getName
jstring
js_name=(jstring)env->CallObjectMethod(customer,methodId,NULL);
...
}
3.c向java返回对象类型
在c程序中首先要创建要返回的java对象,得到每个属性的id,然后给每个属性赋值,最后返回。举例说明:同样是customer对象,有name等属性值,需要在c程序中给每个属性赋值后返回。
JNIEXPORT jobject JNICALL
Java_com_oracle_estt_sc_db_impl_SCQueryODBC__1getCustomer
(JNIEnv *env,
jobject, jobject customer){
......
//发现java Customer类,如果失败,程序返回
jclass clazz =
env->FindClass("com/oracle/estt/sc/busi/Customer");
if(clazz == 0)
return 0;
//为新的java类对象obj分配内存
jobject obj =
env->AllocObject(clazz);
//发现类中的属性,如果失败,程序返回
jfieldID
fid_id = env->GetFieldID(clazz,"customerID","I");
if (fid_id == 0)
return 0;
jfieldID fid_name =
env->GetFieldID(clazz,"name","Ljava/lang/String;");
if (fid_name ==
0)
return 0;
......
env->SetIntField(obj, fid_id, 1
env->SetObjectField(obj, fid_name,
jname);
......
return obj;
}
4.c向java传递一个含有java对象的数组
对于这种情况,先得到数组的大小,接下来取出数组中的对象,取得对象的属性值或者调用对象的方法,将获得值存到本地数组中,然后可以灵活使用这些数据了。举例说明:java向c传递一个含有多个customer对象的数组,在c中将这个数组的分解出来,存到本地的临时数组中去。
JNIEXPORT void JNICALL
Java_com_oracle_estt_sc_db_impl_SCInsertODBC__1insertCustomeRequest___3Lcom_oracle_estt_sc_busi_CustomerRequest_2
(JNIEnv
*env, jobject, jobjectArray oa){
......
//声明customerrequest对象
jobject o_customer;
int i;
jmethodID methodId;
jint
size=env->GetArrayLength(oa);
_tmp_bind[0]= (char *)malloc(size*sizeof(int));
_tmp_bind[1]= (char
*)malloc(size*sizeof(char)*( 20 + 1));
...
//将输入数组的数据拷贝到临时数组中去
for(i=0;iGetObjectArrayElement(oa,i);
//获得customerrequest对象的句柄
jclass
cls_objClass=env->GetObjectClass(o_request);
//获得customerrequest对象的特定方法getCustomerID的id
methodId=env->GetMethodID(cls_objClass,"getCustomerID","()I");
//调用customerrequest对象的特定方法getCustomerID
int_customerID=env->CallIntMethod(o_request,methodId,NULL);
//获得customerrequest对象中特定方法getTelNum的id
methodId=env->GetMethodID(cls_objClass,"getTelNum","()Ljava/lang/String;");
//调用customerrequest对象的特定方法getTelNum
str_telNum=(jstring)env->CallObjectMethod(o_request,methodId,NULL);
...
//将用户id拷贝到临时数组
memcpy(_tmp_bind[0]+i*sizeof(int),&int_customerID,sizeof(int));
//将电话号码拷贝到临时数组,如果电话号码字符串超长,报错返回
if(sizeof(char)*strlen(chr_tel)NewObjectArray(MAX_LINE,env->FindClass("com/oracle/estt/sc/busi/CustomerRequest"),
0); jobject obj;
//发现java Customerrequest类,如果失败,程序返回
jclass clazz =
env->FindClass("com/oracle/estt/sc/busi/CustomerRequest");
if(clazz ==
0)
return 0;
while ((rc = SQLFetch(hstmt)) == SQL_SUCCESS ||rc == SQL_SUCCESS_WITH_INFO)
{
obj = env->AllocObject(clazz);
jfieldID fid_customerID = env->GetFieldID(clazz,"customerID","I");
if (fid_customerID == 0)
return 0;
jfieldID fid_priority = env->GetFieldID(clazz,"priority","I");
if
(fid_priority == 0)
return 0;
...
env->SetIntField(obj, fid_customerID, col_customerID);
env->SetIntField(obj, fid_priority, col_priority);
...
//将对象obj添加到object
array中
if(jSetObjectArrayElement(jo_array, j,
obj);
}else{
break;
}
}
return jo_array;
}
6.jstring向char* 的转换
jstring不能直接在c程序中使用,需要转换成char*。重要的一点是,在使用完char*之后,一定要记得将其释放,以免发生内存泄漏。如下例:
JNIEXPORT jobjectArray JNICALL
Java_com_oracle_estt_sc_db_impl_SCQueryODBC__1getCustomerRequestByCondition
(JNIEnv
*env, jobject, jstring condition, jint customerid){
//将jstring转换为cha*
char* str_condition=(char*)
env->GetStringUTFChars(condition,JNI_FALSE);
......
//释放变量
env->ReleaseStringUTFChars(condition,str_condition);
......
}
7.char*转换成jstring
这个转换就比较麻烦了,但是在数据库操作时会用到。比如,从数据库查询得到的是char*,但是给对象属性赋值的时候需要用jstring,这是需要用到这种转换。具体如下例:
char* col_timestamp=.....;
//加载string类
jclass strClass =
env->FindClass("Ljava/lang/String;");
//获得方法id
jmethodID ctorID =
env->GetMethodID(strClass, "", "([BLjava/lang/String;)V");
//将字符串转换为jstring
bytes_time =
env->NewByteArray(strlen(col_timestamp));
env->SetByteArrayRegion(bytes_time,
0, strlen(col_timestamp), (jbyte*)col_timestamp);
jstring js_time =
env->NewStringUTF("utf-8");
js_time=(jstring)env->NewObject(strClass, ctorID, bytes_time, js_time)
8.java类的原型获取方法
在c中创建java对象和调用java对象方法时需要用到java类的原型,特别是其方法签名。具体办法是:到java类所在的目录下,键入名命令:
>javap -s -p 包路径.java类名
以上几点是我这两天写jni程序的一点总结,写出来与大家分享,欢迎批评指导。
分享到:
相关推荐
JNI开发Java和C/C++互相传递List集合, 可以参考: Java从C/C++获取List集合对象:https://blog.csdn.net/niuba123456/article/details/80994166 Java传递List集合对象到C/C++ ...
通过jni完成java调用c/c++,包含c/c++生成Dll动态库
1、搞音视频必须有扎实的C/C++开发语言基础、 JNI基础,学会如何交叉编译 FFmpeg 等 C/C++库; 2、其次,我们需要阅读大量的音视频规范/协议文档(如H264、MP4/FLV、RTP/RTCP等)。这些文档中的内容基本都以位为单位...
android C/C++ JNI普通数据类型 测试工程, 最好配合文章看《Android JNI C/C++编程--数据类型》
使用C/C++实现Java的Native方法接口(JNI)/ JNI编程(C/C++) 代码实例
本教程主要讲解java中如何调用C/C++,C/C++中如何调用java,并带Demo和调试指南
C/C++中访问Java方法 C/C++中访问Java父类的方法 C/C++中访问/修改Java变量 Java中访问C/C++方法 Java中访问/修改C/C++变量 动态方式实现: C/C++中访问Java方法 C/C++中访问Java父类的方法 C/C++中访问/修改Java...
java调用C/C++全过程 JAVA以其跨平台的特性深受人们喜爱,而又正由于它的跨平台的目的,使得它和本地机器的各种内部联系变得很少,约束了它的功能。解决JAVA对本地操作的一种方法就是JNI。 JAVA通过JNI调用本地...
内含有 可直接使用jar包 、java源码、C/C++源码 纯java编写G711、仅支持alaw与pcm互转 :https://download.csdn.net/download/qq_41054313/10976718 ---------------------------------------------以下下内容为更新...
Android JNI/NDK开发(2)JNI实现C/C++与Android/JAVA相互调用 http://blog.csdn.net/u014702653/article/details/71141423
V. Eclipse+CDT+MinGW 进行JAVA调用C/C++ 34 一、 安装eclipse3.2。 34 二、 安装MinGW。 34 1、 下载MinGW 34 2、 安装 34 3、 安装版本 34 4、 选择安装的编译器 34 5、 选择安装路径,下一步 35 6、 等待下载...
上篇文章中已经介绍了,关于NDK开发环境的搭建,这里不做赘述。这篇文章主要是通过一个例子来说明如何通过eclispe自动生成.h头文件,适合初学者,详见http://write.blog.csdn.net/postedit/42144847
标签:github、luben、zstd、jni、中文文档、jar包、java; 使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请...
标签:github、luben、zstd、jni、中英对照文档、jar包、java; 使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译...
cocos2d-x 通过JNI实现c/c++和Android的java层函数互调, 本文主要实现两个功能: (1)通过Android sdk的API得到应用程序的包名(PackageName),然后传递给c++层函数。 (2)通过c++函数调用Android的java层函数,显示一...
如果你想知道怎样用java去本地调用C/C++接口,本文档应该是你需要的,特别对于无基础的朋友,无疑是很好的JNI入门指南,希望对你有帮助。
标签:github、luben、zstd、jni、中英对照文档、jar包、java; 使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译...
cocos2d-x 通过JNI实现c/c++和Android的java层函数互调-源码 详情请移步到:http://codingnow.cn/program/992.html
Android Studio3.0开发JNI流程,C++调用Java以及Java调用C++,C++/Java互相调用一个简单案例
标签:github、luben、zstd、jni、中英对照文档、jar包、java; 使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译...