亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

C++和JNI的數據轉換

系統 2580 0

如何使用JNI的一些基本方法和過程在網上多如牛毛,如果你對Jni不甚了解,不知道Jni是做什么的,如何建立一個基本的jni程序,或許可以參考下面下面這些文章:
利用VC++6.0實現JNI的最簡單的例子
JNI入門教程之HelloWorld篇
SUN JNI Tutorial

這 些資料的例子中,大多數只是輸入一些簡單的參數,獲取沒有參數。而在實際的使用過程中,往往需要對參數進行處理轉換。才可以被C/C++程序識別。比如我 們在C++中有一個結構(Struct)DiskInfo ,需要傳遞一個類似于DiskInfo *pDiskInfo的參數,類似于在C++這樣參數如何傳遞到Java中呢?下面我們就來討論C++到Java中方法的一些常見參數的轉換:

定義Native Java類:

如果你習慣了使用JNI,你就不會覺得它難了。既然本地方法是由其他語言實現的,它們在Java中沒有函數體。但是,所有本地代碼必須用本地關鍵詞聲明,成為Java類的成員。假設我們在C++中有這么一個結構,它用來描述硬盤信息:

// 硬盤信息
struct {
char name[ 256 ];
int serial;
}
DiskInfo;

那么我們需要在Java中定義一個類來與之匹配,聲明可以寫成這樣:

class DiskInfo {
// 名字
public Stringname;

// 序列號
public int serial;
}

在這個類中,申明一些Native的本地方法,來測試方法參數的傳遞,分別定義了一些函數,用來傳遞結構或者結構數組,具體定義如下面代碼:

/**/ /* *****************定義本地方法******************* */
// 輸入常用的數值類型(Boolean,Byte,Char,Short,Int,Float,Double)
public native void displayParms(StringshowText, int i,booleanbl);

// 調用一個靜態方法
public native int add( int a, int b);

// 輸入一個數組
public native void setArray(boolean[]blList);

// 返回一個字符串數組
public nativeString[]getStringArray();

// 返回一個結構
public nativeDiskInfogetStruct();

// 返回一個結構數組
public nativeDiskInfo[]getStructArray();

編譯生成C/C++頭文件

定義好了Java類之后,接下來就要寫本地代碼。本地方法符號提供一個滿足約定的頭文件,使用 Java工具Javah 可以很容易地創建它而不用手動去創建。你對Java的class文件使用javah命令,就會為你生成一個對應的C/C++頭文件。
1、在控制臺下進入工作路徑,本工程路徑為:E:\work\java\workspace\JavaJni。
2、運行javah 命令:javah -classpath E:\work\java\workspace\JavaJni com.sundy.jnidemo ChangeMethodFromJni
本文生成的C/C++頭文件名為: com_sundy_jnidemo_ChangeMethodFromJni.h


在C/C++中實現本地方法

生成C/C++頭文件之后,你就需要寫頭文件對應的本地方法。注意:所有的本地方法的第一個參數都是指向 JNIEnv 結構的。這個結構是用來調用JNI函數的。第二個參數jclass的意義,要看方法是不是靜態的(static)或者實例(Instance)的。前者,jclass代表一個類對象的引用,而后者是被調用的方法所屬對象的引用。

返回值和參數類型 根據等價約定映射到本地C/C++類型,如表 JNI類型映射 所示。有些類型,在本地代碼中可直接使用,而其他類型只有通過JNI調用操作。

表A

Java 類型 本地類型 描述
boolean jboolean C/C++8位整型
byte jbyte C/C++帶符號的8位整型
char jchar C/C++無符號的16位整型
short jshort C/C++帶符號的16位整型
int jint C/C++帶符號的32位整型
long jlong C/C++帶符號的64位整型e
float jfloat C/C++32位浮點型
double jdouble C/C++64位浮點型
Object jobject 任何Java對象,或者沒有對應java類型的對象
Class jclass Class對象
String jstring 字符串對象
Object[] jobjectArray 任何對象的數組
boolean[] jbooleanArray 布爾型數組
byte[] jbyteArray 比特型數組
char[] jcharArray 字符型數組
short[] jshortArray 短整型數組
int[] jintArray 整型數組
long[] jlongArray 長整型數組
float[] jfloatArray 浮點型數組
double[] jdoubleArray 雙浮點型數組

JNI類型映射

使用數組:

JNI通過JNIEnv提供的操作Java數組的功能。它提供了兩個函數:一個是操作java的簡單型數組的,另一個是操作對象類型數組的。

因為速度的原因,簡單類型的數組作為指向本地類型的指針暴露給本地代碼。因此,它們能作為常規的數組存取。這個指針是指向實際的Java數組或者Java數組的拷貝的指針。另外,數組的布置保證匹配本地類型。

為了存取Java簡單類型的數組,你就要要使用 GetXXXArrayElements 函數(見 B ),XXX代表了數組的類型。這個函數把Java數組看成參數,返回一個指向對應的本地類型的數組的指針。

表B

函數 Java 數組類型 本地類型
GetBooleanArrayElements jbooleanArray jboolean
GetByteArrayElements jbyteArray jbyte
GetCharArrayElements jcharArray jchar
GetShortArrayElements jshortArray jshort
GetIntArrayElements jintArray jint
GetLongArrayElements jlongArray jlong
GetFloatArrayElements jfloatArray jfloat
GetDoubleArrayElements jdoubleArray jdouble

JNI數組存取函數

當你對數組的存取完成后,要確保調用相應的 ReleaseXXXArrayElements 函數,參數是對應Java數組和 GetXXXArrayElements 返回的指針。如果必要的話,這個釋放函數會復制你做的任何變化(這樣它們就反射到java數組),然后釋放所有相關的資源。

為了使用java對象的數組,你必須使用 GetObjectArrayElement 函數和 SetObjectArrayElement 函數,分別去get,set數組的元素。 GetArrayLength 函數會返回數組的長度。

使用對象

JNI提供的另外一個功能是在本地代碼中使用Java對象。通過使用合適的JNI函數,你可以創建 Java對象,get、set 靜態(static)和實例(instance)的域,調用靜態(static)和實例(instance)函數。JNI通過ID識別域和方法,一個域或 方法的ID是任何處理域和方法的函數的必須參數。

表C 列出了用以得到靜態(static)和實例(instance)的域與方法的JNI函數。每個函數接受(作為參數)域或方法的類,它們的名稱,符號和它們對應返回的 jfieldID jmethodID 。

表C

函數 描述
GetFieldID 得到一個實例的域的ID
GetStaticFieldID 得到一個靜態的域的ID
GetMethodID 得到一個實例的方法的ID
GetStaticMethodID 得到一個靜態方法的ID

※域和方法的函數

如果你有了一個類的實例,它就可以通過方法 GetObjectClass 得到,或者如果你沒有這個類的實例,可以通過 FindClass 得到。符號是從域的類型或者方法的參數,返回值得到字符串,如 D 所示。

表D

Java 類型 符號
boolean Z
byte B
char C
short S
int I
long L
float F
double D
void V
objects對象 Lfully-qualified-class-name;L類名
Arrays數組 [array-type [數組類型
methods方法 (argument-types)return-type(參數類型)返回類型

※確定域和方法的符號


下面我們來看看,如果通過使用數組和對象,從C++中的獲取到Java中的DiskInfo 類對象,并返回一個DiskInfo數組:

// 返回一個結構數組,返回一個硬盤信息的結構數組
JNIEXPORTjobjectArrayJNICALLJava_com_sundy_jnidemo_ChangeMethodFromJni_getStructArray
(JNIEnv
* env,jobject_obj)
{
// 申明一個object數組
jobjectArrayargs = 0 ;

// 數組大小
jsizelen = 5 ;

// 獲取object所屬類,一般為ava/lang/Object就可以了
jclassobjClass = (env) -> FindClass( " java/lang/Object " );

// 新建object數組
args = (env) -> NewObjectArray(len,objClass, 0 );

/**/ /* 下面為獲取到Java中對應的實例類中的變量 */

// 獲取Java中的實例類
jclassobjectClass = (env) -> FindClass( " com/sundy/jnidemo/DiskInfo " );

// 獲取類中每一個變量的定義
// 名字
jfieldIDstr = (env) -> GetFieldID(objectClass, " name " , " Ljava/lang/String; " );
// 序列號
jfieldIDival = (env) -> GetFieldID(objectClass, " serial " , " I " );

// 給每一個實例的變量付值,并且將實例作為一個object,添加到objcet數組中
for ( int i = 0 ;i < len;i ++ )
{
// 給每一個實例的變量付值
jstringjstr = WindowsTojstring(env, " 我的磁盤名字是D: " );
// (env)->SetObjectField(_obj,str,(env)->NewStringUTF("mynameisD:"));
(env) -> SetObjectField(_obj,str,jstr);
(env)
-> SetShortField(_obj,ival, 10 );

// 添加到objcet數組中
(env) -> SetObjectArrayElement(args,i,_obj);
}

// 返回object數組
return args;

}



全部的C/C++方法實現代碼如下:

/**/ /*
*
*一縷陽光(sundy)版權所有,保留所有權利。
*/

/**/ /* *
*
*TODOJni中一個從Java到C/C++參數傳遞測試類
*
*@author劉正偉(sundy)
*@see
http://www.cnweblog.com/sundy
*@seemailto:sundy26@126.com
*@version1.0
*@since2005-4-30
*
*修改記錄:
*
*日期修改人描述
*----------------------------------------------------------------------------------------------
*
*
*
*/

// JniManage.cpp:定義DLL應用程序的入口點。
//
packagecom.sundy.jnidemo;
#include
" stdafx.h "

#include
< stdio.h >
#include
< math.h >
#include
" jni.h "
#include
" jni_md.h "

#include
" ./head/Base.h "
#include
" head/wmi.h "
#include
" head/com_sundy_jnidemo_ChangeMethodFromJni.h " // 通過javah–jnijavactransfer生成
#include < stdio.h >
#include
" stdlib.h "
#include
" string.h "

#pragmacomment(lib,
" BaseInfo.lib " )
#pragmacomment(lib,
" jvm.lib " )
// 硬盤信息
struct {
char name[ 256 ];
int serial;
}
DiskInfo;
/**/ /* BOOLAPIENTRYDllMain(HANDLEhModule,
DWORDul_reason_for_call,
LPVOIDlpReserved
)
{
LPTSTRstrName=newCHAR[256];
(*GetHostName)(strName);
printf("%s\n",strName);
delete[]strName;

returnTRUE;
}
*/

// 將jstring類型轉換成windows類型
char * jstringToWindows(JNIEnv * env,jstringjstr);
// 將windows類型轉換成jstring類型
jstringWindowsTojstring(JNIEnv * env, char * str);

// 主函數
BOOLWINAPIDllMain(HANDLEhHandle,DWORDdwReason,LPVOIDlpReserved)
{
return TRUE;
}

// 輸入常用的數值類型Boolean,Byte,Char,Short,Int,Float,Double
JNIEXPORT void JNICALLJava_com_sundy_jnidemo_ChangeMethodFromJni_displayParms
(JNIEnv
* env,jobjectobj,jstrings,jinti,jbooleanb)
{
const char * szStr = (env) -> GetStringUTFChars(s, 0 );
printf(
" String=[%s]\n " ,szStr);
printf(
" int=%d\n " ,i);
printf(
" boolean=%s\n " ,(b == JNI_TRUE ? " true " : " false " ));
(env)
-> ReleaseStringUTFChars(s,szStr);
}


// 調用一個靜態方法,只有一個簡單類型輸出
JNIEXPORTjintJNICALLJava_com_sundy_jnidemo_ChangeMethodFromJni_add
(JNIEnv
* env,jobject,jinta,jintb)
{
int rtn = ( int )(a + b);
return (jint)rtn;
}


/**/ /// /輸入一個數組,這里輸入的是一個Boolean類型的數組
JNIEXPORT void JNICALLJava_com_sundy_jnidemo_ChangeMethodFromJni_setArray
(JNIEnv
* env,jobject,jbooleanArrayba)
{
jboolean
* pba = (env) -> GetBooleanArrayElements(ba, 0 );
jsizelen
= (env) -> GetArrayLength(ba);
int i = 0 ;
// changeevenarrayelements
for (i = 0 ;i < len;i += 2 )
{
pba[i]
= JNI_FALSE;
printf(
" boolean=%s\n " ,(pba[i] == JNI_TRUE ? " true " : " false " ));
}

(env)
-> ReleaseBooleanArrayElements(ba,pba, 0 );
}


/**/ /// /返回一個字符串數組
JNIEXPORTjobjectArrayJNICALLJava_com_sundy_jnidemo_ChangeMethodFromJni_getStringArray
(JNIEnv
* env,jobject)
{
jstringstr;
jobjectArrayargs
= 0 ;
jsizelen
= 5 ;
char * sa[] = { " Hello, " , " world! " , " JNI " , " is " , " <spa
分享到:
評論

C++和JNI的數據轉換


更多文章、技術交流、商務合作、聯系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦?。?!

發表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 好吊色青青青国产欧美日韩 | 日本不卡在线一区二区三区视频 | 亚洲高清在线播放 | 天天操夜夜添 | 久久久影院亚洲精品 | 伊人网99 | 日本高清一级做a爱过程免费视频 | 国产一级特黄aa毛片 | 欧美在线一级片 | 国产亚洲精品一品区99热 | 精品成人免费视频 | 久久久久国产精品免费 | 久久综合九色综合精品 | 视频国产91| 99热都是精品| 久久久久久久国产高清 | 亚洲天堂三区 | 国内精品免费一区二区观看 | www.国产一区二区三区 | 日韩成人在线网站 | 豆国产96在线 | 亚洲 | 性视频一区二区三区免费 | 羞羞视频免费网站在线 | 久久精品欧美一区二区 | 中文字幕日韩在线一区国内 | 日韩欧美伊人久久大香线蕉 | 亚洲a级黄色片 | 九九爱精品 | 久久久日本精品一区二区三区 | 久久这里有精品 | 国内精品亚洲 | 日日插天天干 | 久久99热精品免费观看k影院 | 成人精品亚洲 | 精品国产成人综合久久小说 | 青草免费视频 | 一级毛片免费视频日本 | 欧美洲久久日韩欧美 | 欧美影院一区二区三区 | 日本一级毛片片在线播放 | 欧美精品视|