美文网首页
JNI数组操作

JNI数组操作

作者: tsia | 来源:发表于2019-04-14 22:56 被阅读0次

    在JNI中,java的基本数据类型可以直接与jni基本类型映射,但数组作为引用类型不能直接使用和修改,JNI提供了一组访问和处理数组的API。

    创建数组

    使用New<Type>Array函数创建一个数组实例,其中Type为基本数据类型:Boolean、Byte、Char、Short、Int、Long、Float、Double,如NewIntArray

    jintArray array = env->NewIntArray(4);
        if (0 != array) {
            // 内存溢出的情况下,NewIntArray返回NULL
        }
    

    访问和更新数组

    JNI提供了两种访问数组的方法,一种是复制到c/c++数组中,一种是提供指向数组元素的指针。

    方式一:复制到c数组

    比如java的native传入了一个数组,我们在JNI中读取并更新这个数组,然后对应java的数组就改变了。我们通过复制到c数组的方式,其一般调用流程如下:

    1. get array region

    Get<Type>ArrayRegion函数会将给定的基本Java数组复制到C数组中

    2. update c array

    使用和修改c数组

    3. set array region

    当我们想把修改提交到提交给JNI数组从而改变对应的java数组时,可以使用Set<Type>ArrayRegion函数

    示例

    示例程序中传入了一个java float数组,我们在jni中通过这种方式改变数组的第二个值,执行完后java数组的第二个值会改变。
    jni:

    extern "C" JNIEXPORT void JNICALL
    Java_smarttime_tsia_com_jnitest3_MainActivity_updateArray(
            JNIEnv* env, jobject obj, jfloatArray jnums) {
        // 获取数组长度
        jsize len = env->GetArrayLength(jnums);
    
        // JNI数组复制到c数组
        jfloat buffer[2];
        env->GetFloatArrayRegion(jnums, 0, len, buffer);
    
        // 更新c数组
        buffer[0] = 2.2f;
    
        // 将修改提交到JNI数组中
        env->SetFloatArrayRegion(jnums, 0, len, buffer);
    }
    

    java:

    ...
    public void onClick(View v) {
    
        float[] nums = new float[]{1.f, 2.f};
        updateArray(nums);
    }
    native void updateArray(float[] nums);
    ...
    

    当数组比较大的时候,可以只复制或更新关心的元素区间,以避免出现的性能问题。

    方式二:直接指针的操作

    一般调用流程:

    1. get array elements

    Get<Type>ArrayElements函数获取指向数组元素的直接指针,这个函数的最后一个参数是isCopy,让调用者确定返回的c指针是指向副本,还是指向堆中的固定对象。

    2. update c array

    使用和修改c数组

    3. release array elements

    使用完直接指针后需要立即释放,通过Release<Type>ArrayElements,否则会出现内存泄露

    示例

    以下代码同样会更新java数组的第二个元素:

    extern "C" JNIEXPORT void JNICALL
    Java_smarttime_tsia_com_jnitest3_MainActivity_updateArray(
            JNIEnv* env, jobject obj, jfloatArray jnums) {
        jboolean isCopy;
        // 获取直接指针
        jfloat *parray = env->GetFloatArrayElements(jnums, &copy);
        if (0==parray) {
            return;
        }
    
        // 更新c数组
        parray[1] = 2.2f;
    
        // 释放数组。注意:释放模式为0,java数组会修改;如果是JNI_ABORT,java的数组并不会被修改
        env->ReleaseFloatArrayElements(jnums, parray, 0);
    }
    

    释放模式

    Release<Type>ArrayElements的最后一个参数mode为释放模式,其值和意义为:

    • 0:c数组修改后,将其复制到jni数组,并释放c数组。
    • JNI_COMMIT:c数组修改后,将其复制到jni数组,但不释放c数组。这种一般用于周期性的更新一个java数组。
    • JNI_ABORT:c数组修改后,不将其复制到jni数组,并释放c数组。也就是上述例子设置JNI_ABORT的话,java的数组并不会被改变。

    JNI_COMMIT不会释放数组,最终都需要调用0/JNI_ABORT参数来释放。

    isCopy参数

    Get<Type>ArrayElements的最后一个参数isCopy会告诉我们指针指向的是拷贝的副本,还是原始数据。
    假如我们需要对一个jni数组临时改变一下,然后给其他方法用,但并不希望提交到java数组。这时我们可以通过isCopy判断如果是false说明是原始数据,那我们就需要拷贝一个副本用来修改,否则会修改java数组;假如是true,就不需要另外拷贝一个副本。

    如果isCopy是false,仍然需要主动调用释放的方法,因为即使没有拷贝副本,原始数据也不会自动回收的。

    参考:《Android c++高级编程》

    相关文章

      网友评论

          本文标题:JNI数组操作

          本文链接:https://www.haomeiwen.com/subject/ywqewqtx.html