日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

jni详解

發布時間:2024/9/30 编程问答 23 豆豆
生活随笔 收集整理的這篇文章主要介紹了 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程序的一點總結,寫出來與大家分享,歡迎批評指導。

總結

以上是生活随笔為你收集整理的jni详解的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。