程序员人生 网站导航

JNI/NDK开发指南(五)――访问数组(基本类型数组与对象数组)

栏目:php教程时间:2015-01-14 08:59:22


          转载请注明出处: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就会挂掉。

------分隔线----------------------------
------分隔线----------------------------

最新技术推荐