`

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

    博客分类:
  • c++
阅读更多

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程序的一点总结,写出来与大家分享,欢迎批评指导。


 
分享到:
评论
1 楼 songfantasy 2011-08-18  
不错,学习了

相关推荐

    JNI开发Java和C/C++互相传递List集合

    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++互相调用

    JNI(Java Native Interface)是Java平台的标准组成部分,它允许Java代码和其他语言写的代码进行交互。...通过深入学习这些材料,开发者可以熟练地在Java和C/C++之间进行互操作,实现更复杂的功能。

    使用C/C++实现Java的Native方法接口(JNI)/ JNI编程(C/C++) 代码实例

    JNI提供了大量API用于在Java和C/C++之间交互,如创建和销毁Java对象、调用方法、访问字段、处理数组、异常处理等。例如: - `env->FindClass(className)`:查找Java类。 - `env->GetMethodID(cls, methodName, ...

    jni java调用c/c++

    Java Native Interface (JNI) 是 Java虚拟机(JVM)提供的一种接口,允许 Java 代码和本地代码(如 C/C++)之间的交互。通过 JNI,Java 程序可以调用 C/C++ 代码,实现 Java 与 native 代码的交互。 在本例中,我们...

    使用JNI进行混合编程:在Java中调用C/C++本地库

    在这里,你可以调用任何C/C++代码,与Java对象交互。 - 编译C/C++代码,生成动态链接库。例如,在Windows上,你可以使用`cl Sample1.cpp /Fe:Sample1.dll /LD`。 3. **Java端加载本地库**: - 在Java代码中,我们...

    android NDK入门篇----混合使用java和c/c++代码 源码工程

    总结来说,这个入门项目教你如何设置一个基本的Android NDK环境,创建Java和C/C++之间的交互,并最终生成一个可以在Android设备上使用的本地库。通过这种方式,开发者可以充分利用C/C++的性能优势,同时保持Java的...

    C++库封装JNI接口-实现java调用c++

    这个头文件包含了Java方法的C/C++声明,供C++代码使用。 3. **编写C++代码**:在C++中实现本地方法。引入生成的头文件,并实现其中的方法。例如: ```cpp #include "MyNativeClass.h" JNIEXPORT void JNICALL ...

    JNI技术手册 c/c++调用java

    JNI是Java与本地代码之间沟通的桥梁,使得Java应用程序能够调用本地库中的方法,同时也支持从本地代码调用Java方法。 #### 二、Java调用C/C++ ##### 1. 制作DLL文件 **第一步:** 创建一个C/C++ DLL文件,通常是...

    JNI DEMO:java jni技术 调用 c/c++ 的dll

    "native.jar"是一个Java类库,可能包含了Java层与C/C++交互的类,这些类声明了native方法并负责加载DLL。在运行Java应用时,需要确保环境变量`java.library.path`包含了DLL文件的位置,这样Java虚拟机才能找到并加载...

    java调用C/C++过程

    Java调用C/C++的过程,通常被称为JNI(Java Native Interface),是Java平台标准的一部分,它允许Java代码和其他语言写的代码进行交互...希望这个教程能为初学者提供一个良好的起点,进一步探索Java与C/C++之间的交互。

    java调用C/C++全过程

    通过上述步骤,我们可以实现Java与C/C++之间的交互,这对于需要高性能计算或特定硬件访问的应用来说非常有用。虽然这个过程相对复杂,但掌握后能够极大地扩展Java应用程序的能力。希望本篇文章能帮助开发者更好地...

    JAVA调用C/C++ DLL文件方法

    JAVA 调用 C/C++ 库文件(DLL)是 Java 语言与 C/C++ 语言之间的交互方式之一。在 Java 中,存在多种调用 C/C++ 库文件的方式,包括 JNI、JNative、Jawin 和 Jacob 等。其中,JNI(Java Native Interface)是 Java ...

    Android-JNI完整工程,实现c++调用java和java调用c++

    - JNI接口提供了Java虚拟机(JVM)和本地代码之间的桥梁,使得开发者可以使用Java调用C/C++函数,反之亦然。 - JNI头文件(`jni.h`)包含了定义了JNI函数和数据类型的声明,是编写JNI代码的基础。 2. **创建JNI...

    linux-java调用c语言编译的so动态库-jni例子-简单计算器

    这个例子对于理解JNI的工作原理和Java与C/C++之间的交互非常有帮助。它展示了如何在跨语言环境中实现高效计算,同时保持Java的平台独立性。通过学习和实践这样的例子,开发者可以更好地理解和运用JNI技术,解决复杂...

    jni.zip_c调用java_java 调用 c++_安卓 JNI

    在Android平台上,JNI(Java Native Interface)是一种关键的技术,它允许Java代码与本地(C/C++)代码进行交互。JNI是Java平台标准的一部分,旨在提供一个安全、高效的方式来调用和被调用,使得开发者可以利用C/C++...

    zstd-jni-1.4.3-1-API文档-中文版.zip

    标签:github、luben、zstd、jni、中文文档、jar包、java; 使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请...

    Java 调用C/C++程序

    3. 编写C/C++源代码,实现头文件中的函数原型,使用JNI函数进行Java对象和数据类型的转换。 4. 编译C/C++源代码为动态链接库。 5. 在Java程序中使用`System.loadLibrary()`加载动态链接库。 6. 在Java方法中调用对应...

    Hello-jni-ndk

    在Android开发中,JNI通常用于调用C/C++库,以实现高性能计算、图形处理或者利用系统底层功能,而"Hello-jni-ndk"就是一个基础的JNI示例项目,旨在演示Java和C/C++代码之间的互调用。 1. **JNI基础知识** - JNI...

    java与c++交互(JNI学习笔记)

    在Java Native Interface (JNI) 中,Java 和 C/C++ 的数据类型有着明确的对应关系。理解这些对应关系是实现Java与C++交互的基础。 - **基本类型**: - Java 的 `boolean` 对应于 C/C++ 的 `jboolean` 类型,通常是...

Global site tag (gtag.js) - Google Analytics