-3

为什么在 jni 端做一个 foo 循环,在 java 端再做一个?我不可能只返回没有 for 循环的数组吗?

JNIEXPORT jobjectArray JNICALL Java_main_returndays(JNIEnv *env, jobject jobj)
{

  char *days[]={"Sunday",
                "Monday",
                "Tuesday",
                "Wednesday",
                "Thursday",
                "Friday",
                "Saturday"};

  jstring str;
  jobjectArray day = 0;
  jsize len = 7;
  int i;

  day = (*env)->NewObjectArray(env,len,(*env)->FindClass(env,"java/lang/String"),0);

  for(i=0;i<7;i++)//// without this loop ?
  {
    str = (*env)->NewStringUTF(env,days[i]);
    (*env)->SetObjectArrayElement(env,day,i,str);
  }

return day;
}
4

1 回答 1

1

为了通过 JNI 将 String[] 返回给 Java,您必须创建数组,然后将每个字符串转换并设置到数组中。没有任何捷径可以像将 char** 传回 Java。

但是...如果您的目标是避免通过 JNI 在 C++ 中创建 Java 字符串,而是在 Java 中进行,您可以完成此操作。有时这样做有充分的理由。在我们的软件中,我们将复杂的数据结构从 C++ 处理到 Java。我们发现使用 JNI 构建数据结构非常慢(每个 JNI 方法调用就像 1000 个时钟周期)。在 C++ 中序列化为字节数组通常更快,创建单个 JNI 字节数组,将其跨 JNI 边界交给 Java,然后让 Java 反序列化字节数组。

我不会给你一个完整的、有效的例子,但它是这样的。我将在 C++ 中构建一个字节数组,每个字符串之间都有空值,然后将其交给 Java 处理成一个列表(比 String[] 更容易)。

C++:

// Find overall length with nulls
int len = 0;
for (int i = 0; i < sizeof(days)/sizeof(char*); i++) {
   len += strlen(days[i]) + 1;
}
// Fill C++ byte array
char* cBytes = new char[len];
char* ptr = cBytes;
for (int i = 0; i < sizeof(days)/sizeof(char*); i++) {
   strcpy(ptr, days[i]); 
   ptr += strlen(days[i]) + 1;
}
// Copy C++ to Java
jbyteArray jBytes = env->NewByteArray(len);
env->SetByteArrayRegion(jBytes, 0, len, cBytes);
delete [] cBytes;
return jBytes;

爪哇:

byte jBytes[] = // call C++ via JNI
List<String> strs = new ArrayList<>();
int offset = 0;
// break byte array at null, convert to String
while (offset < jBytes.length) {
   for (int i = offset; true; i++) {
      if (jBytes[i] == 0) {
         // Default code page assumed.  You could also pass a Charset.
         strs.add(new String(jBytes, offset, i - offset));
         offset = i + 1;
         break;
      }
   }
}

我意识到在 C++ 和 Java 之间传递数组有“更优化”的方式,通常涉及某种 NativeByteBuffer 包装器。但这些方法往往有特殊情况,比较复杂。

于 2017-05-14T14:18:26.597 回答