zoukankan      html  css  js  c++  java
  • 访问数组(JNI)

    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; 
     } 
  • 相关阅读:
    Win(Phone)10开发第(7)弹,Extended Execution
    Win(Phone)10开发第(5)弹,本地媒体服务器的一些注意事项
    Win(Phone)10开发第(4)弹,HTTP 实时流播放 m3u8
    Win(Phone)10开发第(3)弹,简单的Demo程序网络请求json解析列表显示
    Win(Phone)10开发第(2)弹,导出APPX包并签名部署
    Win(Phone)10开发第(1)弹,桌面和手机的扩展API,还我后退键
    WP8里dll类库(SDK)实现多语言多主题
    Windows Phone中解决多模块多程序集之间相互循环引用的问题一种思路
    在继承中子类自动实现单例模式
    ansible---基础
  • 原文地址:https://www.cnblogs.com/lijunamneg/p/2828922.html
Copyright © 2011-2022 走看看