JNI: 访问对象中的数组

需求是这样的:

需要在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: 访问对象中的数组

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