問鼎國(guó)產(chǎn)手機(jī)華為Mate 10 Pro評(píng)測(cè)
java與c/c++之間的數(shù)據(jù)交互—–jni點(diǎn)滴
最近作一個(gè)TIemsten數(shù)據(jù)庫(kù)的項(xiàng)目,用到了jni技術(shù)。在這個(gè)項(xiàng)目中,我們用java來(lái)寫界面和業(yè)務(wù)邏輯,用c語(yǔ)言寫數(shù)據(jù)庫(kù)odbc訪問。單純的odbc其實(shí)沒有什么難的,但是在java和c之間進(jìn)行數(shù)據(jù)傳遞是比較麻煩的事情。兩者之間數(shù)據(jù)的傳遞有這樣幾種情況:java和c之間基本數(shù)據(jù)類型的交互,java向c傳遞對(duì)象類型,c向java返回對(duì)象類型,c調(diào)用java類。下面就這樣幾種情況分類說明。
1、java 向c傳遞基本數(shù)據(jù)類型
對(duì)于基本數(shù)據(jù)類型,java和c是相互對(duì)應(yīng)的,所以可以直接使用。它們的對(duì)應(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傳遞對(duì)象類型
對(duì)于java傳遞進(jìn)來(lái)的java對(duì)象模型,c要加載java類的原型,根據(jù)創(chuàng)建相應(yīng)的c對(duì)象,獲取java對(duì)象的方法的id,然后調(diào)用java對(duì)象的方法。舉例說明:比如有個(gè)java類customer對(duì)象作為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對(duì)象的句柄
jclass cls_objClass=env->GetObjectClass(customer);
//獲得customer對(duì)象中特定方法getName的id
methodId=env->GetMethodID(cls_objClass,”getName”,”()Ljava/lang/String;”);
//調(diào)用customer對(duì)象的特定方法getName
jstring js_name=(jstring)env->CallObjectMethod(customer,methodId,NULL);
…
}
3.c向java返回對(duì)象類型
在c程序中首先要?jiǎng)?chuàng)建要返回的java對(duì)象,得到每個(gè)屬性的id,然后給每個(gè)屬性賦值,最后返回。舉例說明:同樣是customer對(duì)象,有name等屬性值,需要在c程序中給每個(gè)屬性賦值后返回。
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類對(duì)象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傳遞一個(gè)含有java對(duì)象的數(shù)組
對(duì)于這種情況,先得到數(shù)組的大小,接下來(lái)取出數(shù)組中的對(duì)象,取得對(duì)象的屬性值或者調(diào)用對(duì)象的方法,將獲得值存到本地?cái)?shù)組中,然后可以靈活使用這些數(shù)據(jù)了。舉例說明:java向c傳遞一個(gè)含有多個(gè)customer對(duì)象的數(shù)組,在c中將這個(gè)數(shù)組的分解出來(lái),存到本地的臨時(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對(duì)象
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í)數(shù)組中去
for(i=0;i