java与c/c++之间的数据交互-----jni点滴

淡泊明志、宁静致远

A Diamond is just a piece of Coal that did well under Pressure.

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;i<size;i++){
 //从数组中获得customerrequest对象
    o_request=env->GetObjectArrayElement(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)<=sizeof(char)*( 20 + 1)){
 memcpy(_tmp_bind[1]+i*sizeof(char)*( 20+1 ),chr_tel,strlen(chr_tel)+1);
 }else{
 printf("%s too long!\n",chr_tel);
 return;
 }

...

}

...

}

5.c向java返回一个数组

先创建数组,然后加载java对象,给每个java对象的属性赋值,添加到数组中,最后返回数组。如下例:

JNIEXPORT jobjectArray JNICALL Java_com_oracle_estt_sc_db_impl_SCQueryODBC__1getCustomerRequest
(JNIEnv *env, jobject, jint customerid){

......

//声明存放查询结果的objectarray
 jobjectArray jo_array = env->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(j<MAX_LINE){
   env->SetObjectArrayElement(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, "<init>", "([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程序的一点总结,写出来与大家分享,欢迎批评指导。

推荐资料:

Java Native Interface Specification.http://java.sun.com/j2se/1.5.0/docs/guide/jni/spec/jniTOC.html

时间: 2024-12-13 03:09:34

java与c/c++之间的数据交互-----jni点滴的相关文章

java不同对象之间的数据交互(通用)

??java中万物皆对象.不论你是线程,还是异步任务,还是...都可以用以下的三种通用方法来进行对象间数据的交互.当然android相比java有自己独特的数据交互方式,这些会在后面的文章中讲到,本篇文章意在对java通用的对象间的数据交互进行总结. 一.通过构造函数传递 ??构造函数很大程度上是方便参数的传递,以达到在新建对象的时候,同时对这个对象的一些属性进行初始化.我们经常需要根据自己的需要,重载类的构造方法.我们可以在对象初建的时候把它需要的数据传递给它,以实现不同对象之间数据的交互.

Android技术10:Java与C语言之间简单数据的传递

由于C语言和Java语言之间很多类型不一致性,因此使用native时,需要数据类型转换.下面演示分别传递整型,字符串,整型数组,Java静态方法传递数据. 1.创建native方法 我们单独创建一个NativeClass类来存放native方法 1 package com.forsta.ndk; 2 3 public class NativeClass { 4 public native int add(int x,int y); 5 public native String showStrin

Android进阶之Fragment与Activity之间的数据交互

1 为什么 因为Fragment和Activity一样是具有生命周期,不是一般的bean通过构造函数传值,会造成异常. 2 Activity把值传递给Fragment 2.1 第一种方式,也是最常用的方式,就是使用Bundle来传递参数 (1)宿主Activity/FragmentAdapter中: Bundle bundle = new Bundle(); bundle.putString(Constant.INTENT_ID, productId); Fragment fragment =

Java之基于S2SH与手机数据交互(一)

在前两篇博客,介绍了在eclipse上搭建SSH,可是好多小伙伴反映.看了偶写滴博客.跟着搭建还是错误百出,唉! 事实上不经历错误怎么能不见红线啊!于是我在上篇博客补充了他们的错误,还在被错误困扰的童鞋,能够去看一看. 今天我们简介一下.怎样在SSH基础上与手机进行数据交互.如今后台给移动端传递数据一般都是以json的格式传递给移动端,然后移动端在对json进行解析.说白了就是给一个链接,打开之后就像下图那样的数据. 像这种数据该怎么给手机端提供呢?接下来就来学习一下怎样给移动端提供数据. (一

UWP开发:APP之间的数据交互(以微信为例)

目录 说明 UWP应用唤醒方式 跟微信APP交互数据 APP之间交互数据的前提 说明 我们经常看到,在手机上不需要退到桌面,APP之间就可以相互切换,并且可以传递数据.比如我在使用知乎APP的时候,需要使用新浪微博账号登录,点击“微博登录”后,系统自动唤醒新浪微博APP,并将知乎请求登录的数据传给了微博APP(微博APP界面上可以看到是知乎请求登录授权),见下图:    如上所示,依次点击“微博登录”.“确定”,手机界面自动来回切换,不需要人为干预.另外,如果我们在知乎看到一篇有意思的文章,需要

html与js和php之间实现数据交互

<div class="top3"> <input id="KeyWord" type="text" class="form1" name="keyWord" value="请输入关键字" onfocus="javascript:if(this.value=='请输入关键字')this.value='';"> <input type=&q

#游戏平台接入#Android游戏平台接入(一)#android客户端和C++底层的数据交互#jni编程#欢迎交流#

1.jni是什么 2.jni应用情景 3.java 调用C++过程解析(附例子) 4.C++调用java 过程解析(附例子) 5.java,c++,jni数据类型对照 6.常用jni函数解析 http://blog.csdn.net/skywalker256/article/details/4677644

Vue2.0子同级组件之间数据交互

接着我们进入Demo,首先我们可以删除掉模板项目中src/components/Hello.vue,然后在App.vue中删除对于Hello子组件的注册和使用还有一些其他无关紧要的东西,此时的App.vue应为这样 一 .我们先来创建中央事件总线,在src/assets/下创建一个eventBus.js,内容如下(eventBus中我们只创建了一个新的Vue实例,以后它就承担起了组件之间通信的桥梁了,也就是中央事件总线.) 二 . 创建一个firstChild组件,引入eventBus这个事件总

困扰几周了,请教啊,android与websevice数据交互很诡异的问题

============问题描述============ 传输数据(title,imgurl,account)当传输title或title+account时数据正常传输,但是无法传输Imgurl项,即使imgurl+title都一样,没有任何数据显示 请教 1.httpcon.java复制内容到剪贴板代码: //数据交互,输入输出流与读取流 import java.io.IOException; import java.io.InputStream; import java.io.OutputS