需求是这样的:
需要在C++层做一些操作,然后返回给java层一个对象, 返回给java层的对象结构大概如下:
class JavaClass {
...
int i;
byte[] a;
}
将C++层计算得到的int 和byte[]结果赋值给java层的对象,
赋值int成员很容易,无非就是:
jclass jcClass = env->GetObjectClass(jc);
jfieldID iId = env->GetFieldID(jcClass, "i", "I");
// This way we can get and set the "i" field. Let's double it:
jint i = env->GetIntField(jc, iId);
env->SetIntField(jc, iId, i * 2);
// The jfieldID of the "a" field (byte array) can be got like this:
jfieldID aId = env->GetFieldID(jcClass, "a", "[B");
巴拉巴拉,最终调用下env->SetIntField()就完事了。可给byte[] 数组赋值就麻烦了, 一时间没有找到对应的JNI方法,原以为会写成 env->SetByteArrayField()这样的,结果就是没有:
看了半天,除了8种基本的数据类型和对象类型,以及它们对应的static 类型,愣是没找到给数组类型的成员赋值的函数。。。
不过在stack overflow上找到一种解决办法:
JNIEXPORT void JNICALL Java_Test_process(JNIEnv * env, jclass c, jobject jc) {
jclass jcClass = env->GetObjectClass(jc);
jfieldID iId = env->GetFieldID(jcClass, "i", "I");
// This way we can get and set the "i" field. Let's double it:
jint i = env->GetIntField(jc, iId);
env->SetIntField(jc, iId, i * 2);
// The jfieldID of the "a" field (byte array) can be got like this:
jfieldID aId = env->GetFieldID(jcClass, "a", "[B");
// Get the object field, returns JObject (because Array is instance of Object)
jobject mvdata = env->GetObjectField (jc, aID);
// Cast it to a jdoublearray
jdoubleArray * arr = reinterpret_cast<jdoubleArray*>(&mvdata)
// Get the elements (you probably have to fetch the length of the array as well
double * data = env->GetDoubleArrayElements(*arr, NULL);
// Don't forget to release it
env->ReleaseDoubleArrayElements(*arr, data, 0);
}
主要是看标红的地方(它这里double数组),它这里调用的是取对象域的方法 GetObjectField , 然后取jobject的地址,将结果强转成jdoubleArray, 这样一来问题就解决了,一旦取到了jdoubleArray, 后面就可以直接使用 env->SetDoubleArrayRegion 方法来给数组赋值了,不得不说这想法确实很妙!
其实主要还是没想到原来在JNI里面数组是被当做对象来处理的。