需求是这样的:

需要在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()这样的,结果就是没有:

JNI: 访问对象中的数组_java

看了半天,除了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里面数组是被当做对象来处理的。