需求是这样的:
需要在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上找到一种解决办法:
How to access arrays within an object with JNI?
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里面数组是被当做对象来处理的。