Android JNI知识总结

来源:互联网 发布:mac 之间airdrop怎么用 编辑:程序博客网 时间:2024/06/06 06:45

在JNI中进行java的数组操作

JNI在处理基本类型数组和对象数组上面是不同的。对象数组里面是一些指向对象实例或者其它数组的引用。 

基本类型数组:

获取数组元素指针的对应关系:
  函数            数组类型
  GetBooleanArrayElements   boolean
   GetByteArrayElements    byte
  GetCharArrayElements     char
  GetShortArrayElements    short
  GetIntArrayElements     int
   GetLongArrayElements    long
  GetFloatArrayElements     float
  GetDoubleArrayElements   double
  
释放数组元素指针的对应关系:
  Function            Array Type
   ReleaseBooleanArrayElements   boolean
   ReleaseByteArrayElements    byte
   ReleaseCharArrayElements    char
   ReleaseShortArrayElements    short
   ReleaseIntArrayElements     int
   ReleaseLongArrayElements    long
   ReleaseFloatArrayElements    float
   ReleaseDoubleArrayElements   double

看一个简单的例子。下面的程序调用了一个本地方法 sumArray,这个方法对一个 int数组里面的元素进行累加:

复制代码
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 < 10; i++) {              arr[i] = i;          }          int sum = p.sumArray(arr);          System.out.println("sum = " + sum);      }      static {          System.loadLibrary("IntArray");      }  } 
复制代码

数组的引用类型是一般是jarray或者jarray的子类型jintArray。就像jstring不是一个C字符串类型一样,jarray也不是一个C数组类型。
所以,不要直接访问 jarray。你必须使用合适的JNI函数来访问基本数组元素:

使用GetIntArrayRegion 函数来把一个 int数组中的所有元素复制到一个C缓冲区中,然后我们在本地代码中通过C缓冲区来访问这些元素。

复制代码
JNIEXPORT jint JNICALL  Java_IntArray_sumArray(JNIEnv *env, jobject obj, jintArray arr) {      jint buf[10];      jint i, sum = 0;      (*env)->GetIntArrayRegion(env, arr, 0, 10, buf);          for (i = 0; i < 10; i++) {          sum += buf[i];      }      return sum; } 
复制代码

 

JNI支持一系列的Get/Release<Type>ArrayElement 函数,这些函数允许本地代码获取一个指向基本类型数组的元素的指针。

复制代码
JNIEXPORT jint JNICALL  Java_IntArray_sumArray(JNIEnv *env, jobject obj, jintArray arr)  {      jint *carr;      jint i, sum = 0;      carr = (*env)->GetIntArrayElements(env, arr, NULL); //推荐使用     if (carr == NULL) {          return 0; /* exception occurred */      }      for (i=0; i<10; i++) {          sum += carr[i];      }      (*env)->ReleaseIntArrayElements(env, arr, carr, 0);      return sum;  }
复制代码

如果你想在一个预先分配的C缓冲区和内存之间交换数据,应该使用Get/Set</Type>ArrayRegion系列函数。这些函数会进行越界检查,在需要的时候会有可能抛出ArrayIndexOutOfBoundsException异常。 
对于少量的、固定大小的数组,Get/Set<Type>ArrayRegion是最好的选择,因为C缓冲区可以在Stack(栈)上被很快地分配,而且复制少量数组元素的代价是很小的。这对函数的另外一个优点就是,允许你通过传入一个索引和长度来实现对子字符串的操作。

如果你没有一个预先分配的 C 缓冲区,并且原始数组长度未定,而本地代码又不想在获取数组元素的指针时阻塞的话,使用 Get/ReleasePrimitiveArrayCritical 函数对。就像Get/ReleaseStringCritical函数对一样,这对函数很小心地使用,以避免死锁。

Get/Release<type>ArrayElements 系列函数永远是安全的。JVM 会选择性地返回一个指针,这个指针可能指向原始数据也可能指向原始数据复制。

 

对象数组:
JNI提供了一个函数对来访问对象数组。GetObjectArrayElement返回数组中指定位置的元素,而SetObjectArrayElement修改数组中指定位置的元素。
与基本类型的数组不同的是,你不能一次得到所有的对象元素或者一次复制多个对象元素。
字符串和数组都是引用类型,你要使用Get/SetObjectArrayElement来访问字符串数组或者数组的数组。

下面的例子调用了一个本地方法来创建一个二维的 int数组,然后打印这个数组的内容:



class ObjectArrayTest {      private static native int[][] initInt2DArray(int size);      public static void main(String[] args) {          int[][] i2arr = initInt2DArray(3);          for (int i = 0; i < 3; i++) {              for (int j = 0; j < 3; j++) {                   System.out.print(" " + i2arr[i][j]);              }              System.out.println();          }      }      static {          System.loadLibrary("ObjectArrayTest");      }  } 


静态本地方法 initInt2DArray 创建了一个给定大小的二维数组。执行分配和初始化数组任务的本地方法可以是下面这样子的:



JNIEXPORT jobjectArray JNICALL Java_ObjectArrayTest_initInt2DArray(JNIEnv *env, jclass cls, int size) {      jobjectArray result;      int i;      jclass intArrCls = (*env)->FindClass(env, "[I");      if (intArrCls == NULL) {          return NULL; /* exception thrown */      }           result = (*env)->NewObjectArray(env, size, intArrCls,  NULL); //分配第一维          if (result == NULL) {          return NULL; /* out of memory error thrown */      }           for (i = 0; i < size; i++) {          jint tmp[256];  /* make sure it is large enough! */          int j;          jintArray iarr = (*env)->NewIntArray(env, size); //创建第二维数据         if (iarr == NULL) {              return NULL; /* out of memory error thrown */          }          for (j = 0; j < size; j++) {              tmp[j] = i + j;          }          (*env)->SetIntArrayRegion(env, iarr, 0, size, tmp);          (*env)->SetObjectArrayElement(env, result, i, iarr);          (*env)->DeleteLocalRef(env, iarr);      }      return result;  } 
军事评论
0 0