Warning: file_get_contents(/data/phpspider/zhask/data//catemap/4/c/67.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
如何将字符数组作为字节[]从C JNI函数传递给Java方法_Java_C_Arrays_Java Native Interface_Argument Passing - Fatal编程技术网

如何将字符数组作为字节[]从C JNI函数传递给Java方法

如何将字符数组作为字节[]从C JNI函数传递给Java方法,java,c,arrays,java-native-interface,argument-passing,Java,C,Arrays,Java Native Interface,Argument Passing,我很难找到将char缓冲区从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)

我很难找到将char缓冲区从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
是这里使用的正确函数吗?传递给它的正确内容是什么,如何分配它,以及应该如何(如果有的话)释放它


代码片段可能是最简洁的答案,用几句话解释对象的所有权是如何发生的。

您正在寻找的函数是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;
}

下面的示例用于将char[]从C代码传递到Java字节[]

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));
}

我认为您必须将GetMethodID参数类型更改为“([C)V”。@Yohji No,
C
是Java
char
,它是16位,代表一个unicode字符。C
char
通常代表一个本机字节(在本例中假定为8位)(但在这里不是)用于表示任何8位编码的8位字符。@hyde Java
char
拥有一个UTF-16代码单元;其中一个或两个表示Unicode字符。提供了一个简洁的解释。为什么不使用buf_len?我还需要将数组的长度传递给Java?对于本例来说,实际上不需要UTF-16代码单元,我可能忘记删除它e如果从我复制的东西粘贴了这个片段。你将需要它来处理C Java转换。我不确定Java溢出buf会发生什么。我怀疑如果没有设置isCopy,你会得到null,如果是指针,isCopy的JNI_为TRUE。下面的ShivBuy有一个很好的例子,说明了buf_len与NewBy一起需要在哪里泪滴()。