【问题标题】:How to pass char array from C JNI function to Java method as byte[]如何将 C JNI 函数中的 char 数组作为字节 [] 传递给 Java 方法
【发布时间】:2014-07-27 09:59:50
【问题描述】:

我无法找到将字符缓冲区从 JNI 方法传递到 Java 方法的正确文档。这是代码

jint JNICALL Java_foo_package_MyJavaClass_myNativeMethod(JNIEnv *jenv, jobject jobj)
{
    jclass clazz = (*jenv)->GetObjectClass(jenv, jobj);
    //  MyJavaClass method:  private void addData(byte[] data)
    jmethodID mid = (*jenv)->GetMethodID(jenv, clazz, "addData", "([B)V");
    assert(mid);

    const char buf[] = { 0, 1, 2, 3, 42 };
    const size_t buf_len = sizeof buf;

    (*jenv)->CallVoidMethod(jenv, jobj, mid, buf /* obviously wrong */ );

    return 0;
}

CallVoidMethod 是在这里使用的正确函数吗,传递给它的正确内容是什么,如何分配它,以及应该如何(如果有的话)释放它?

代码 sn-p 可能是最简洁的答案,用几句话解释对象的所有权是如何变化的。

【问题讨论】:

  • 我认为您必须将 GetMethodID 参数类型更改为“([C)V”。
  • @Yohji 不,C 是 Java char,它是 16 位的,代表一个 unicode 字符。 C char 表示一个本地字节(在这种情况下假定为 8 位),通常(但不是这里)用于表示任何 8 位编码的 8 位字符。
  • @hyde Java char 拥有一个 UTF-16 代码单元;其中一个或两个代表一个 Unicode 字符。 Character.toChars(int) 提供了简洁的解释。

标签: java c arrays java-native-interface argument-passing


【解决方案1】:

您要查找的函数是 GetByteArrayElements 和 ReleaseByteArrayElements。

这样的事情应该可以解决问题:

jint JNICALL Java_foo_package_MyJavaClass_myNativeMethod(JNIEnv *jenv, jobject jobj)
{
    jclass clazz = (*jenv)->GetObjectClass(jenv, jobj);
    //  MyJavaClass method:  private void addData(byte[] data)
    jmethodID mid = (*jenv)->GetMethodID(jenv, clazz, "addData", "([B)V");
    assert(mid);

    const char buf[] = { 0, 1, 2, 3, 42 };
    const size_t buf_len = sizeof buf;

    jboolean isCopy;
    jbyte *jbuf = (*jenv)->GetByteArrayElements(jenv, buf, &isCopy);

    (*jenv)->CallVoidMethod(jenv, jobj, mid, jbuf);

    (*jenv)->ReleaseByteArrayElements(jenv, buf, jbuf, 0);

    return JNI_OK;
}

【讨论】:

  • 为什么不使用buf_len?我也需要将数组的长度传递给 Java?
  • Is 对于这个例子来说并不是真正需要的,如果从我复制的任何内容中粘贴了这个 sn-p,我可能忘记删除。您将需要它来处理 C Java 转换。我不确定java溢出buf会发生什么。如果 isCopy 未设置,我怀疑你会返回 null ,如果 isCopy 是指针,则返回 JNI_TRUE 。下面 ShivBuy 有一个很好的例子,说明在哪里需要 buf_len 和 NewByteArray()。
【解决方案2】:

以下示例适用于将 char[] 从 C 代码传递到 Java byte[]。

void JNICALL Java_com_example_testapplication_MainActivity_getJNIByteArrayArg(JNIEnv    *jenv, jobject jobj)
{
jclass clazz = (*jenv)->FindClass(jenv, "com/example/testapplication/MainActivity"); // class path
jmethodID mid = (*jenv)->GetMethodID(jenv, clazz, "addData", "([B)V");// function name

jbyteArray retArray;
char data[] = {'a','b',3,4,5};
int data_size = 5;
if(!retArray)
retArray = (*jenv)->NewByteArray(jenv, data_size);

if((*jenv)->GetArrayLength(jenv, retArray) != data_size)
{
    (*jenv)->DeleteLocalRef(jenv, retArray);
    retArray = (*jenv)->NewByteArray(jenv, data_size);
}

void *temp = (*jenv)->GetPrimitiveArrayCritical(jenv, (jarray)retArray, 0);
memcpy(temp, data, data_size);
(*jenv)->ReleasePrimitiveArrayCritical(jenv, retArray, temp, 0);

(*jenv)->CallVoidMethod(jenv, jobj, mid, retArray);
}
public void addData(byte[] data) {
    System.out.println("Buyya: From C: " + new String(data));
}

【讨论】:

    猜你喜欢
    • 2016-09-22
    • 1970-01-01
    • 1970-01-01
    • 2016-07-21
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多