将字节数组从本机代码传递到Java端

时间:2012-03-01 15:23:45

标签: java android-ndk java-native-interface

我的目标是将一个可变长度的字节数组从本机代码传递到Java端。 Java类实例方法将bytearray作为其参数:

private void writeBuffer(final byte[] buffer)
{
}

我能够在本机代码中找到方法ID:

jclass cls = (*env)->FindClass(env,"class_path");
jmethodID writeBufferMethodID = (*env)->GetMethodID(env, cls, "writeBuffer", "([B)V");

但仍然无法弄清楚如何正确传递字节数组。我试过了:

jbyteArray retArray = (*env)->NewByteArray(env, data_size);
void *temp = (*env)->GetPrimitiveArrayCritical(env, (jarray)retArray, 0);
memcpy(temp, decoded_frame->data[0], data_size);
(*env)->CallVoidMethod(env, obj, writeBufferMethodID, retArray);
(*env)->ReleasePrimitiveArrayCritical(env, retArray, temp, 0);

还有:

retArray = (*env)->NewByteArray(env, data_size);
(*env)->SetByteArrayRegion(env, retArray, 0, data_size, (jbyte *)decoded_frame->data[0]);
(*env)->CallVoidMethod(env, obj, writeBufferMethodID, retArray);

调用Java方法,但一段时间后应用程序崩溃。此外,我收到的Java缓冲区中的所有值都等于零,因此看起来内容根本不会被复制。

我通过将缓冲区内容写入二进制文件来验证本机端的缓冲区内容(decoding_frame-> data [0])并且没有问题,该文件包含了我期望的内容。

我定期打电话给那个方法; 每次调用中,数组的大小可以变化

正确且最强有效方式是什么?在每次调用期间分配一个新数组显然是一个愚蠢的想法,但我不知道,如果数组大小不同,如何避免这种情况。

修改

我已经用这种方式重写了我的代码,现在似乎还可以。

在while循环中调用:

...做一些解码......

if(!retArray)
retArray = (*env)->NewByteArray(env, data_size);

if((*env)->GetArrayLength(env, retArray) != data_size)
{
(*env)->DeleteLocalRef(env, retArray);
retArray = (*env)->NewByteArray(env, data_size);
}

void *temp = (*env)->GetPrimitiveArrayCritical(env, (jarray)retArray, 0);
memcpy(temp, decoded_frame->data[0], data_size);
(*env)->CallVoidMethod(env, obj, writeBufferMethodID, retArray);
(*env)->ReleasePrimitiveArrayCritical(env, retArray, temp, 0);

在循环退出处调用:

(*env)->DeleteLocalRef(env, retArray);

3 个答案:

答案 0 :(得分:5)

在调用CallVoidMethod之前尝试调用ReleasePrimitiveArrayCritical。

答案 1 :(得分:3)

下面的示例适用于将char []从C代码传递到Java byte []。

void JNICALL Java_com_example_testapplication_MainActivity_getJNIByteArrayArg(JNIEnv    *jenv, jobject jobj)
{
jclass clazz = (*jenv)->FindClass(jenv, "com/example/testapplication/MainActivity"); // class path
jmethodID mid = (*jenv)->GetMethodID(jenv, clazz, "addData", "([B)V");// function name

jbyteArray retArray;
char data[] = {'a','b',3,4,5};
int data_size = 5;
if(!retArray)
retArray = (*jenv)->NewByteArray(jenv, data_size);

if((*jenv)->GetArrayLength(jenv, retArray) != data_size)
{
    (*jenv)->DeleteLocalRef(jenv, retArray);
    retArray = (*jenv)->NewByteArray(jenv, data_size);
}

void *temp = (*jenv)->GetPrimitiveArrayCritical(jenv, (jarray)retArray, 0);
memcpy(temp, data, data_size);
(*jenv)->ReleasePrimitiveArrayCritical(jenv, retArray, temp, 0);

(*jenv)->CallVoidMethod(jenv, jobj, mid, retArray);
}
public void addData(byte[] data) {
System.out.println("Buyya: From C: " + new String(data));
}

答案 2 :(得分:0)

您还可以使用SetByteArrayRegion设置jbyteArray内容:

(*env)->SetByteArrayRegion(env, retArray, 0, data_size, decoded_frame->data[0]);