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;
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
与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;
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
原生代码可以像使用普通的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;
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
2.2对直接指针的操作
原生代码可以用GetArrayElements函数获取指向数组元素的直接指针。第三个参数和我们在上篇博文中提到的一样,它是一个可选参数,该可选参数的名称为isCopy,让调用者确定返回的C字符串地址指向副本还是指向堆中的固定对象。
因为可以像普通的C数组一样访问和处理数组元素,因此JNI没有提供访问和处理元素的方法,JNI要求原生代码用完这些指针必须立刻释放,否则会出现内存溢出问题,原生代码可以使用JNI提供的ReleaseArrayElements函数来释放GetArrayElements函数返回的C数组。该函数有四个参数,第四个是释放模式,有以下三种。
0:将内容复制回来并释放原生数组
JNI_COMMIT:将内容复制回来但是不释放原生数组,一般用于周期性的更新一个Java数组。
JNI_ABORT释放原生数组但是不将内容复制回来。
- 1
- 2
- 3
这次我们用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;
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
好了,关于数组操作就说这么多,有问题欢迎留言讨论。
文章来源: wangsong.blog.csdn.net,作者:_江南一点雨,版权归原作者所有,如需转载,请联系作者。
原文链接:wangsong.blog.csdn.net/article/details/48212121
- 点赞
- 收藏
- 关注作者
评论(0)