Как вернуть массив из JNI в Java?
Я пытаюсь использовать Android NDK.
Есть ли способ вернуть массив (в моем случае an int[]
), созданный в JNI, в Java? Если да, пожалуйста, приведите краткий пример функции JNI, которая могла бы это сделать.
-Спасибо
Переведено автоматически
Ответ 1
Если вы изучили документацию и у вас все еще есть вопросы, которые должны быть частью вашего первоначального вопроса. В этом случае функция JNI в примере создает несколько массивов. Внешний массив состоит из массива 'Object', созданного с помощью функции JNI NewObjectArray()
. С точки зрения JNI, это все, что представляет собой двумерный массив, массив объектов, содержащий ряд других внутренних массивов.
Следующий цикл for создает внутренние массивы типа int[] с использованием функции JNI NewIntArray()
. Если бы вы просто хотели вернуть одномерный массив целых чисел, то для создания возвращаемого значения вы бы использовали функцию NewIntArray()
. Если бы вы хотели создать одномерный массив строк, то вы бы использовали NewObjectArray()
функцию, но с другим параметром для класса.
Поскольку вы хотите вернуть массив int, то ваш код будет выглядеть примерно так:
JNIEXPORT jintArray JNICALL Java_ArrayTest_initIntArray(JNIEnv *env, jclass cls, int size)
{
jintArray result;
result = (*env)->NewIntArray(env, size);
if (result == NULL) {
return NULL; /* out of memory error thrown */
}
int i;
// fill a temp structure to use to populate the java int array
jint fill[size];
for (i = 0; i < size; i++) {
fill[i] = 0; // put whatever logic you want to populate the values here.
}
// move from the temp structure to the java structure
(*env)->SetIntArrayRegion(env, result, 0, size, fill);
return result;
}
Ответ 2
если кто-то хотел бы знать, как вернуть String[] array:
код Java
private native String[] data();
собственный экспорт
JNIEXPORT jobjectArray JNICALL Java_example_data() (JNIEnv *, jobject);
машинный код
JNIEXPORT jobjectArray JNICALL
Java_example_data
(JNIEnv *env, jobject jobj){
jobjectArray ret;
int i;
char *message[5]= {"first",
"second",
"third",
"fourth",
"fifth"};
ret= (jobjectArray)env->NewObjectArray(5,
env->FindClass("java/lang/String"),
env->NewStringUTF(""));
for(i=0;i<5;i++) {
env->SetObjectArrayElement(
ret,i,env->NewStringUTF(message[i]));
}
return(ret);
}
по ссылке:
http://www.coderanch.com/t/326467/java/java/Returning-String-array-program-Java
Ответ 3
Исходя из заданного вопроса, это уже объяснялось в первом ответе о том, как мы можем передать int[] через jobjectArray. Но вот пример, как мы можем вернуть jobjectArray, который содержит списки данных. Это может быть полезно, например, в ситуациях: когда кому-то нужно вернуть данные в формате 2D, чтобы нарисовать некоторую линию с точками x и y. Приведенный ниже пример показывает, как jobjectArray может возвращать данные в виде следующего формата:
Ввод Java в JNI:
Массив [Arraylist
из x точек с плавающей запятой][Arraylist
из y точек с плавающей запятой]
Вывод JNI в java:jobjectArray
[Arraylist
из x точек с плавающей запятой] [Arraylist
из y точек с плавающей запятой]
extern "C" JNIEXPORT jobjectArray JNICALL
_MainActivity_callOpenCVFn(
JNIEnv *env, jobject /* this */,
jobjectArray list) {
//Finding arrayList class and float class(2 lists , one x and another is y)
static jclass arrayListCls = static_cast<jclass>(env->NewGlobalRef(env->FindClass("java/util/ArrayList")));
jclass floatCls = env->FindClass("java/lang/Float");
//env initialization of list object and float
static jmethodID listConstructor = env->GetMethodID(arrayListCls, "<init>", "(I)V");
jmethodID alGetId = env->GetMethodID(arrayListCls, "get", "(I)Ljava/lang/Object;");
jmethodID alSizeId = env->GetMethodID(arrayListCls, "size", "()I");
static jmethodID addElementToList = env->GetMethodID(arrayListCls, "add", "(Ljava/lang/Object;)Z");
jmethodID floatConstructor = env->GetMethodID( floatCls, "<init>", "(F)V");
jmethodID floatId = env->GetMethodID(floatCls,"floatValue", "()F");
//null check(if null then return)
if (arrayListCls == nullptr || floatCls == nullptr) {
return 0;
}
// Get the value of each Float list object in the array
jsize length = env->GetArrayLength(list);
//If empty
if (length < 1) {
env->DeleteLocalRef(arrayListCls);
env->DeleteLocalRef(floatCls);
return 0;
}
// Creating an output jObjectArray
jobjectArray outJNIArray = env->NewObjectArray(length, arrayListCls, 0);
//taking list of X and Y points object at the time of return
jobject xPoint,yPoint,xReturnObject,yReturnObject;
//getting the xList,yList object from the array
jobject xObjFloatList = env->GetObjectArrayElement(list, 0);
jobject yObjFloatList = env->GetObjectArrayElement(list, 1);
// number of elements present in the array object
int xPointCounts = static_cast<int>(env->CallIntMethod(xObjFloatList, alSizeId));
static jfloat xReturn, yReturn;
jobject xReturnArrayList = env->NewObject(arrayListCls,listConstructor,0);
jobject yReturnArrayList = env->NewObject(arrayListCls,listConstructor,0);
for (int j = 0; j < xPointCounts; j++) {
//Getting the x points from the x object list in the array
xPoint = env->CallObjectMethod(xObjFloatList, alGetId, j);
//Getting the y points from the y object list in the array
yPoint = env->CallObjectMethod(yObjFloatList, alGetId, j);
//Returning jobjectArray(Here I am returning the same x and points I am receiving from java side, just to show how to make the returning `jobjectArray`)
//float x and y values
xReturn =static_cast<jfloat >(env->CallFloatMethod(xPoint, floatId,j));
yReturn =static_cast<jfloat >(env->CallFloatMethod(yPoint, floatId,j));
xReturnObject = env->NewObject(floatCls,floatConstructor,xReturn);
yReturnObject = env->NewObject(floatCls,floatConstructor,yReturn);
env->CallBooleanMethod(xReturnArrayList,addElementToList,xReturnObject);
env->CallBooleanMethod(yReturnArrayList,addElementToList,yReturnObject);
env->SetObjectArrayElement(outJNIArray,0,xReturnArrayList);
env->SetObjectArrayElement(outJNIArray,1,yReturnArrayList);
__android_log_print(ANDROID_LOG_ERROR, "List of X and Y are saved in the array","%d", 3);
}
return outJNIArray;
Ответ 4
Simple solution is that write the array data in a file from C,and then access the file from Java