JNI:在java和c之間進行數據傳遞

在java和c之間進行數據傳遞,兩者之間數據的傳遞有這樣幾種情況:java和c之間基本數據類型 的交互,java向c傳遞對象類型,c向java返回對象類型,c調用java類。下面就這樣幾種情況分類說明。


  • 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
  • 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);

        ...

    }
  • 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;

    }
  • 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;
                }

                ...

            }

            ...

    }
  • 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;

    }
  • 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);

    ......

    }
  • 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)
  • java類的原型獲取方法
    在c中創建java對象和調用java對象方法時需要用到java類的原型,特別是其方法簽名。具體辦法是:到java類所在的目錄下,鍵入名命令:

javap -s -p 包路徑.java類名

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章