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