Моей целью является передача массива байтов переменной длины из нативного кода на сторону 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]) на нативной стороне, записав их в бинарный файл, и проблем нет, файл содержит именно то, что я ожидаю.
Я периодически вызываю этот метод; размер массива может изменяться в каждом вызове.
Какой правильный и эффективный способ? Выделять новый массив при каждом вызове явно глупая идея, но я не знаю, как этого избежать, если размер массива меняется.
РЕДАКТИРОВАТЬ:
Я переписал свой код таким образом, и теперь все в порядке.
вызывается в цикле 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);