java與c/c++之間的數(shù)據(jù)交互—–jni點滴
最近作一個TIemsten數(shù)據(jù)庫的項目,用到了jni技術(shù)。在這個項目中,我們用java來寫界面和業(yè)務(wù)邏輯,用c語言寫數(shù)據(jù)庫odbc訪問。單純的odbc其實沒有什么難的,但是在java和c之間進行數(shù)據(jù)傳遞是比較麻煩的事情。兩者之間數(shù)據(jù)的傳遞有這樣幾種情況:java和c之間基本數(shù)據(jù)類型的交互,java向c傳遞對象類型,c向java返回對象類型,c調(diào)用java類。下面就這樣幾種情況分類說明。
1、java 向c傳遞基本數(shù)據(jù)類型
對于基本數(shù)據(jù)類型,java和c是相互對應(yīng)的,所以可以直接使用。它們的對應(yīng)關(guān)系為;
Java類型 本地類型 字節(jié)(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類的原型,根據(jù)創(chuàng)建相應(yīng)的c對象,獲取java對象的方法的id,然后調(diào)用java對象的方法。舉例說明:比如有個java類customer對象作為jni參數(shù)傳遞到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;”);
//調(diào)用customer對象的特定方法getName
jstring js_name=(jstring)env->CallObjectMethod(customer,methodId,NULL);
…
}
3.c向java返回對象類型
在c程序中首先要創(chuàng)建要返回的java對象,得到每個屬性的id,然后給每個屬性賦值,最后返回。舉例說明:同樣是customer對象,有name等屬性值,需要在c程序中給每個屬性賦值后返回。
JNIEXPORT jobject JNICALL Java_com_oracle_estt_sc_db_impl_SCQueryODBC__1getCustomer
(JNIEnv *env, jobject, jobject customer){
……
//發(fā)現(xiàn)java Customer類,如果失敗,程序返回
jclass clazz = env->FindClass(“com/oracle/estt/sc/busi/Customer”);
if(clazz == 0)
return 0;
//為新的java類對象obj分配內(nèi)存
jobject obj = env->AllocObject(clazz);
//發(fā)現(xiàn)類中的屬性,如果失敗,程序返回
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對象的數(shù)組
對于這種情況,先得到數(shù)組的大小,接下來取出數(shù)組中的對象,取得對象的屬性值或者調(diào)用對象的方法,將獲得值存到本地數(shù)組中,然后可以靈活使用這些數(shù)據(jù)了。舉例說明:java向c傳遞一個含有多個customer對象的數(shù)組,在c中將這個數(shù)組的分解出來,存到本地的臨時數(shù)組中去。
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));
…
//將輸入數(shù)組的數(shù)據(jù)拷貝到臨時數(shù)組中去
for(i=0;i