多多色-多人伦交性欧美在线观看-多人伦精品一区二区三区视频-多色视频-免费黄色视屏网站-免费黄色在线

國內最全IT社區平臺 聯系我們 | 收藏本站
阿里云優惠2
您當前位置:首頁 > php開源 > php教程 > JNI/NDK開發指南(五)――訪問數組(基本類型數組與對象數組)

JNI/NDK開發指南(五)――訪問數組(基本類型數組與對象數組)

來源:程序員人生   發布時間:2015-01-14 08:59:22 閱讀次數:4219次


          轉載請注明出處:http://blog.csdn.net/xyang81/article/details/42346165


         JNI中的數組分為基本類型數組和對象數組,它們的處理方式是不1樣的,基本類型數組中的所有元素都是JNI的基本數據類型,可以直接訪問。而對象數組中的所有元素是1個類的實例或其它數組的援用,和字符串操作1樣,不能直接訪問Java傳遞給JNI層的數組,必須選擇適合的JNI函數來訪問和設置Java層的數組對象。瀏覽此文假定你已了解了JNI與Java數據類型的映照關系,如果還不了解的童鞋,請移步《JNI/NDK開發指南(3)――JNI數據類型及與Java數據類型的映照關系》瀏覽。下面以int類型為例說明基本數據類型數組的訪問方式,對象數組類型用1個創建2維數組的例子來演示如何訪問:


1、訪問基本類型數組

package com.study.jnilearn; // 訪問基本類型數組 public class IntArray { // 在本地代碼中求數組中所有元素的和 private native int sumArray(int[] arr); public static void main(String[] args) { IntArray p = new IntArray(); int[] arr = new int[10]; for (int i = 0; i < arr.length; i++) { arr[i] = i; } int sum = p.sumArray(arr); System.out.println("sum = " + sum); } static { System.loadLibrary("IntArray"); } }
本地代碼:

/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class com_study_jnilearn_IntArray */ #ifndef _Included_com_study_jnilearn_IntArray #define _Included_com_study_jnilearn_IntArray #ifdef __cplusplus extern "C" { #endif /* * Class: com_study_jnilearn_IntArray * Method: sumArray * Signature: ([I)I */ JNIEXPORT jint JNICALL Java_com_study_jnilearn_IntArray_sumArray (JNIEnv *, jobject, jintArray); #ifdef __cplusplus } #endif #endif // IntArray.c #include "com_study_jnilearn_IntArray.h" #include <string.h> #include <stdlib.h> /* * Class: com_study_jnilearn_IntArray * Method: sumArray * Signature: ([I)I */ JNIEXPORT jint JNICALL Java_com_study_jnilearn_IntArray_sumArray (JNIEnv *env, jobject obj, jintArray j_array) { jint i, sum = 0; jint *c_array; jint arr_len; //1. 獲得數組長度 arr_len = (*env)->GetArrayLength(env,j_array); //2. 根據數組長度和數組元素的數據類型申請寄存java數組元素的緩沖區 c_array = (jint*)malloc(sizeof(jint) * arr_len); //3. 初始化緩沖區 memset(c_array,0,sizeof(jint)*arr_len); printf("arr_len = %d ", arr_len); //4. 拷貝Java數組中的所有元素到緩沖區中 (*env)->GetIntArrayRegion(env,j_array,0,arr_len,c_array); for (i = 0; i < arr_len; i++) { sum += c_array[i]; //5. 累加數組元素的和 } free(c_array); //6. 釋放存儲數組元素的緩沖區 return sum; }
上例中,在Java中定義了1個sumArray的native方法,參數類型是int[],對應JNI中jintArray類型。在本地代碼中,首先通過JNI的GetArrayLength函數獲得數組的長度,已知數組是jintArray類型,可以得出數組的元素類型是jint,然后根據數組的長度和數組元素類型,申請相應大小的緩沖區。如果緩沖區不大的話,固然也能夠直接在棧上申請內存,那樣效力更高,但是沒那末靈活,由于Java數組的大小變了,本地代碼也隨著修改。接著調用GetIntArrayRegion函數將Java數組中的所有元素拷貝到C緩沖區中,并累加數組中所有元素的和,最后釋放存儲java數組元素的C緩沖區,并返回計算結果。GetIntArrayRegion函數第1個參數是JNIEnv函數指針,第2個參數是Java數組對象,第3個參數是拷貝數組的開始索引,第4個參數是拷貝數組的長度,第5個參數是拷貝目的地。下圖是計算結果:

在前面的例子當中,我們通過調用GetIntArrayRegion函數,將int數組中的所有元素拷貝到C臨時緩沖區中,然后在本地代碼中訪問緩沖區中的元夙來實現求和的計算,JNI還提供了1個和GetIntArrayRegion相對應的函SetIntArrayRegion,本地代碼可以通過這個函數來修改所有基本數據類型數組的元素。另外JNI還提供1系列直接獲得數組元素指針的函數Get/Release<Type>ArrayElements,比如:GetIntArrayElements、ReleaseArrayElements、GetFloatArrayElements、ReleaseFloatArrayElements等。下面我們用這類方式重新實現計算數組元素的和:

JNIEXPORT jint JNICALL Java_com_study_jnilearn_IntArray_sumArray2 (JNIEnv *env, jobject obj, jintArray j_array) { jint i, sum = 0; jint *c_array; jint arr_len; // 可能數組中的元素在內存中是不連續的,JVM可能會復制所有原始數據到緩沖區,然后返回這個緩沖區的指針 c_array = (*env)->GetIntArrayElements(env,j_array,NULL); if (c_array == NULL) { return 0; // JVM復制原始數據到緩沖區失敗 } arr_len = (*env)->GetArrayLength(env,j_array); printf("arr_len = %d ", arr_len); for (i = 0; i < arr_len; i++) { sum += c_array[i]; } (*env)->ReleaseIntArrayElements(env,j_array, c_array, 0); // 釋放可能復制的緩沖區 return sum; }
GetIntArrayElements第3個參數表示返回的數組指針是原始數組,還是拷貝原始數據到臨時緩沖區的指針,如果是JNI_TRUE:表示臨時緩沖區數組指針,JNI_FALSE:表示臨時原始數組指針。開發當中,我們其實不關心它從哪里返回的數組指針,這個參數填NULL便可,但在獲得到的指針必須做校驗,由于當原始數據在內存當中不是連續寄存的情況下,JVM會復制所有原始數據到1個臨時緩沖區,并返回這個臨時緩沖區的指針。有可能在申請開辟臨時緩沖區內存空間時,會內存不足致使申請失敗,這時候會返回NULL。
    寫過Java的程序員都知道,在Java中創建的對象全都由GC(垃圾回收器)自動回收,不需要像C/C++1樣需要程序員自己管理內存。GC會實時掃描所有創建的對象是不是還有援用,如果沒有援用則會立即清算掉。當我們創建1個像int數組對象的時候,當我們在本地代碼想去訪問時,發現這個對象正被GC線程占用了,這時候本地代碼會1直處于阻塞狀態,直到等待GC釋放這個對象的鎖以后才能繼續訪問。為了不這類現象的產生,JNI提供了Get/ReleasePrimitiveArrayCritical這對函數,本地代碼在訪問數組對象時會暫停GC線程。不過使用這對函數也有個限制,在Get/ReleasePrimitiveArrayCritical這兩個函數期間不能調用任何會讓線程阻塞或等待JVM中其它線程的本地函數或JNI函數,和處理字符串的Get/ReleaseStringCritical函數限制1樣。這對函數和GetIntArrayElements函數1樣,返回的是數組元素的指針。下面用這類方式重新實現上例中的功能:

JNIEXPORT jint JNICALL Java_com_study_jnilearn_IntArray_sumArray (JNIEnv *env, jobject obj, jintArray j_array) { jint i, sum = 0; jint *c_array; jint arr_len; jboolean isCopy; c_array = (*env)->GetPrimitiveArrayCritical(env,j_array,&isCopy); printf("isCopy: %d ", isCopy); if (c_array == NULL) { return 0; } arr_len = (*env)->GetArrayLength(env,j_array); printf("arr_len = %d ", arr_len); for (i = 0; i < arr_len; i++) { sum += c_array[i]; } (*env)->ReleasePrimitiveArrayCritical(env, j_array, c_array, 0); return sum; }

小結:

1、對小量的、固定大小的數組,應當選擇Get/SetArrayRegion函數來操作數組元素是效力最高的。由于這對函數要求提早分配1個C臨時緩沖區來存儲數組元素,你可以直接在Stack(棧)上或用malloc在堆上來動態申請,固然在棧上申請是最快的。有童鞋可能會認為,訪問數組元素還需要將原始數據全部拷貝1份到臨時緩沖區才能訪問而覺得效力低?我想告知你的是,像這類復制少許數組元素的代價是很小的,幾近可以疏忽。這對函數的另外1個優點就是,允許你傳入1個開始索引和長度來實現對子數組元素的訪問和操作(SetArrayRegion函數可以修改數組),不過傳入的索引和長度不要越界,函數會進行檢查,如果越界了會拋出ArrayIndexOutOfBoundsException異常。

2、如果不想預先分配C緩沖區,并且原始數組長度也不肯定,而本地代碼又不想在獲得數組元素指針時被阻塞的話,使用Get/ReleasePrimitiveArrayCritical函數對,就像Get/ReleaseStringCritical函數對1樣,使用這對函數要非常謹慎,以避免死鎖。

3、Get/Release<type>ArrayElements系列函數永久是安全的,JVM會選擇性的返回1個指針,這個指針可能指向原始數據,也可能指向原始數據的復制。


2、訪問對象數組

       JNI提供了兩個函數來訪問對象數組,GetObjectArrayElement返回數組中指定位置的元素,SetObjectArrayElement修改數組中指定位置的元素。與基本類型不同的是,我們不能1次得到數據中的所有對象元素或1次復制多個對象元素到緩沖區。由于字符串和數組都是援用類型,只能通過Get/SetObjectArrayElement這樣的JNI函數來訪問字符串數組或數組中的數組元素。下面的例子通過調用1個本地方法來創建1個2維的int數組,然后打印這個2維數組的內容:

package com.study.jnilearn; public class ObjectArray { private native int[][] initInt2DArray(int size); public static void main(String[] args) { ObjectArray obj = new ObjectArray(); int[][] arr = obj.initInt2DArray(3); for (int i = 0; i < 3; i++) { for (int j = 0; j < 3; j++) { System.out.format("arr[%d][%d] = %d ", i, j, arr[i][j]); } } } static { System.loadLibrary("ObjectArray"); } }
本地代碼:
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class com_study_jnilearn_ObjectArray */ #ifndef _Included_com_study_jnilearn_ObjectArray #define _Included_com_study_jnilearn_ObjectArray #ifdef __cplusplus extern "C" { #endif /* * Class: com_study_jnilearn_ObjectArray * Method: initInt2DArray * Signature: (I)[[I */ JNIEXPORT jobjectArray JNICALL Java_com_study_jnilearn_ObjectArray_initInt2DArray (JNIEnv *, jobject, jint); #ifdef __cplusplus } #endif #endif // ObjectArray.c #include "com_study_jnilearn_ObjectArray.h" /* * Class: com_study_jnilearn_ObjectArray * Method: initInt2DArray * Signature: (I)[[I */ JNIEXPORT jobjectArray JNICALL Java_com_study_jnilearn_ObjectArray_initInt2DArray (JNIEnv *env, jobject obj, jint size) { jobjectArray result; jclass clsIntArray; jint i,j; // 1.取得1個int型2維數組類的援用 clsIntArray = (*env)->FindClass(env,"[I"); if (clsIntArray == NULL) { return NULL; } // 2.創建1個數組對象(里面每一個元素用clsIntArray表示) result = (*env)->NewObjectArray(env,size,clsIntArray,NULL); if (result == NULL) { return NULL; } // 3.為數組元素賦值 for (i = 0; i < size; ++i) { jint buff[256]; jintArray intArr = (*env)->NewIntArray(env,size); if (intArr == NULL) { return NULL; } for (j = 0; j < size; j++) { buff[j] = i + j; } (*env)->SetIntArrayRegion(env,intArr, 0,size,buff); (*env)->SetObjectArrayElement(env,result, i, intArr); (*env)->DeleteLocalRef(env,intArr); } return result; }
結果:



     本地函數initInt2DArray首先調用JNI函數FindClass取得1個int型的2維數組類的援用,傳遞給FindClass的參數"[I"是JNI class descript(JNI類型描寫符,后面為詳細介紹),它對應著JVM中的int[]類型。如果int[]類加載失敗的話,FindClass會返回NULL,然后拋出1個java.lang.NoClassDefFoundError: [I異常。

     接下來,NewObjectArray創建1個新的數組,這個數組里面的元素類型用intArrCls(int[])類型來表示。函數NewObjectArray只能分配第1維,JVM沒有與多維數組相對應的數據結構,JNI也沒有提供類似的函數來創建2維數組。由于JNI中的2維數組直接操作的是JVM中的數據結構,相比JAVA和C/C++創建2維數組要復雜很多。給2維數組設置數據的方式也非常直接,首先用NewIntArray創建1個JNI的int數組,并為每一個數組元素分配空間,然后用SetIntArrayRegion把buff[]緩沖中的內容復制到新分配的1維數組中去,最后在外層循環中順次將int[]數組賦值到jobjectArray數組中,1維數組中套1維數組,就構成了1個所謂的2維數組。

     另外,為了不在循環內創建大量的JNI局部援用,造成JNI援用表溢出,所以在外層循環中每次都要調用DeleteLocalRef將新創建的jintArray援用從援用表中移除。在JNI中,只有jobject和子類屬于援用變量,會占用援用表的空間,jint,jfloat,jboolean等都是基本類型變量,不會占用援用表空間,即不需要釋放。援用表最大空間為512個,如果超越這個范圍,JVM就會掛掉。

生活不易,碼農辛苦
如果您覺得本網站對您的學習有所幫助,可以手機掃描二維碼進行捐贈
程序員人生
------分隔線----------------------------
分享到:
------分隔線----------------------------
關閉
程序員人生
主站蜘蛛池模板: 图片区 日韩 欧美 亚洲 | 中文字幕第12页 | 欧美日韩在线视频播放 | 中文字幕在线播放量 | 黑人巨大xxxx | 亚洲成在人线av | 巨大乳bbwsex欧美高清 | 国产精品亚洲综合第一区 | 欧美日韩性视频 | 他添的我好湿好爽视频 | 91精品国产99久久 | 在线欧洲成人免费视频 | 久久伊人成人网 | 波霸欧美性猛交xxxxxx | 免费一级做a爰片性色毛片 免费一看一级毛片 | 亚洲国产欧美视频 | 一级欧美一级日韩片 | 欧美国产成人在线 | 亚洲三级图| 五月天婷五月天综合网在线 | 欧美性天天| 久久久久久国产精品视频 | 久久久日本精品一区二区三区 | 久久精品在 | 稀缺资源呦视频在线网站 | 久久精品国产欧美 | 亚洲欧美在线免费观看 | 精品中文字幕不卡在线视频 | 99爱精品 | 日本一区二区视频在线观看 | 国产一区二区在线免费观看 | 中文国产成人精品久久久 | 亚洲第九十九页 | 免费一区二区三区四区 | 2022国产成人精品福利网站 | 在线欧美三级 | 国产精品国产国产aⅴ | 欧美一区二区日韩一区二区 | 欧美做爰gif动态图一区二区 | 国产高清在线播放免费观看 | 99热精品成人免费观看 |