NDK开发之数组操作

JNI把Java数组当作引用类型来处理,JNI提供了必要的函数来访问和处理Java数组。

下面一个一个来看。

1.创建数组
我们可以使用NewArray函数在原生代码中创建数组实例,其中可以是Int、Char、和Boolean等,例如NewIntArray,使用这些函数时应该以参数的形式给出数组的大小。看下面一段代码:

先定义一个jintArray数组,然后定义一个jint数组,将jint数组的内容提交给jintArray,然后返回jintArray。

jintArray Java_com_example_jni_MainActivity_getIntArray(JNIEnv* env,
        jobject thiz) {
    jintArray javaArray;
    jint nativeArr[3] = {21,22,23};
    javaArray = (*env)->NewIntArray(env,3);
    (*env)->SetIntArrayRegion(env,javaArray,0,3,nativeArr);
    return javaArray;
}

与NewString函数一样,在内存溢出的情况下,NewArray函数将返回NULL以通知原生代码虚拟机中有异常抛出,这样原生代码就会停止运行。

2.访问数组元素
JNI提供两种访问Java数组元素的方法,可以将Java数组复制为C数组,或者让JNI提供直接指向数组元素的指针。下面将对这两种方法分别介绍。

2.1对副本的操作
我们可以将一个Java数组复制成一个C数组,然后对这个数组进行操作。
看下面一段代码,Java传进来一个数组,我们计算数组元素之和然后返回。

jint Java_com_example_jni_MainActivity_getSum(JNIEnv* env,
        jobject thiz,jintArray javaArray) {
    //获得Java传递进来数组的长度
    jsize length = (*env)->GetArrayLength(env,javaArray);
    //定义一个C数组
    jint nativeArr[length];
    jint sum = 0;
    //将Java数组区复制到C数组中
    (*env)->GetIntArrayRegion(env,javaArray,0,length,nativeArr);
    int i;
    //求和
    for(i=0;i<length;i++){
        sum+=nativeArr[i];
    }
    return sum;
}

原生代码可以像使用普通的C数组一样使用和修改数组元素。当原生代码相将所做的修改提交给Java数组时,可以使用SetArrayRegion函数将C数组复制回Java数组中。
看下面代码,我们将Java传进来的数组中的每个元素加1,然后再把修改后的数组返回。

jintArray Java_com_example_jni_MainActivity_arrAdd1(JNIEnv* env,
        jobject thiz,jintArray javaArray) {
    //获得Java传递进来数组的长度
    jsize length = (*env)->GetArrayLength(env,javaArray);
    //定义一个C数组
    //将Java数组区复制到C数组中
    jint nativeArr[length];
    (*env)->GetIntArrayRegion(env,javaArray,0,length,nativeArr);
    int i;
    //修改元素的值
    for(i=0;i<length;i++){
        nativeArr[i]=nativeArr[i]+1;
    }
    //从C数组向Java数组提交所做的修改
    (*env)->SetIntArrayRegion(env,javaArray,0,length,nativeArr);
    return javaArray;
}

2.2对直接指针的操作

原生代码可以用GetArrayElements函数获取指向数组元素的直接指针。第三个参数和我们在上篇博文中提到的一样,它是一个可选参数,该可选参数的名称为isCopy,让调用者确定返回的C字符串地址指向副本还是指向堆中的固定对象。

因为可以像普通的C数组一样访问和处理数组元素,因此JNI没有提供访问和处理元素的方法,JNI要求原生代码用完这些指针必须立刻释放,否则会出现内存溢出问题,原生代码可以使用JNI提供的ReleaseArrayElements函数来释放GetArrayElements函数返回的C数组。该函数有四个参数,第四个是释放模式,有以下三种。

0:将内容复制回来并释放原生数组
JNI_COMMIT:将内容复制回来但是不释放原生数组,一般用于周期性的更新一个Java数组。
JNI_ABORT释放原生数组但是不将内容复制回来。

这次我们用C做个简单的乘法,Java传进来一个数组,对该数组中元素求积之后返回。

jint Java_com_example_jni_MainActivity_multiplication(JNIEnv* env,jobject thiz,jintArray javaArray){
    jint* nativeDirectArray;
    jboolean isCopy;
    jint result = 1;
    jsize length = (*env)->GetArrayLength(env,javaArray);
    nativeDirectArray = (*env)->GetIntArrayElements(env,javaArray,&isCopy);
    int i = 0;
    for(i=0;i<length;i++){
        result *= nativeDirectArray[i];
    }
    (*env)->ReleaseIntArrayElements(env,javaArray,nativeDirectArray,0);
    return result;
}

好了,关于数组操作就说这么多,有问题欢迎留言讨论。

你可能感兴趣的:(android,NDK)