Передача байтового массива из нативного кода на сторону Java - PullRequest
12 голосов
/ 01 марта 2012

Моя цель - передать байтовый массив переменной длины из собственного кода на сторону Java.Метод экземпляра класса Java принимает bytearray в качестве аргумента:

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

Я могу найти идентификатор метода в собственном коде:

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, которые я получаю, равны нулю, поэтому кажется, что контент вообще не копируется.

Я проверил содержимое буферов (decoded_frame-> data [0]) на нативной стороне, записав их в двоичный файл, и проблем нет, файл содержит именно то, что я ожидаю.

Я называю этот метод периодически;размер массива может варьироваться в каждый вызов.

Какой правильный и самый эффективный способ?Выделение нового массива во время каждого вызова, очевидно, глупая идея, но я не знаю, как этого избежать, если размер массива меняется.

РЕДАКТИРОВАТЬ:

Iпереписал мой код таким образом, и теперь все в порядке.

вызывается в цикле 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);

Ответы [ 2 ]

5 голосов
/ 01 марта 2012

Попробуйте вызвать ReleasePrimitiveArrayCritical перед вызовом CallVoidMethod.

3 голосов
/ 20 ноября 2014

Ниже приведен пример передачи char [] из кода C в байт Java [].

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));
}
...