我正在用JNI测试直接ByteBuffer(java.nio.ByteBuffer)。因此,下面的代码试图:
我在想我到底做错了什么?C++代码设法从Java中获取数据,但是C++中的变化并没有在Java中反映出来。
这就是我在java上所做的:
public static void main(String[] args){
ByteBuffer bb = ByteBuffer.allocateDirect(3);
byte[] b = {122,121,120};
System.out.println("1: " + new String(b));
bb.put(b);
new JNI.process(bb);
byte[] c = new byte[3];
c[0] = bb.get();
System.out.println("4: " + new String(c));
}这就是我对JNI函数所做的:
JNIEXPORT void JNICALL Java_MarsJNI_mapreduce
(JNIEnv *env, jobject thisObj, jobject output){
char *out = (char*)env->GetDirectBufferAddress(output);
printf("2: %s\n", out);
out = "ABC";
printf("3: %s\n", out);
}我得到的结果是:
1: zyx
2: zyx
3: ABC
Exception in thread "main" java.nio.BufferUnderflowException
at java.nio.Buffer.nextGetIndex(Buffer.java:474)
at java.nio.DirectByteBuffer.get(DirectByteBuffer.java:208)
at MarsJNI.main(MarsJNI.java:21)发布于 2014-03-15 12:14:40
第一个问题:见@TedBigham的答案。您也可以使用buf.rewind()。
第二个问题:您只将缓冲区的第一个字节复制到c中,而不是整个缓冲区。做:
byte[] c = new byte[3];
bb.rewind();
bb.put(c);
System.out.println("4: " + new String(c));第三个问题:您的C++代码:
char *out = (char*)env->GetDirectBufferAddress(output);
// ...
out = "ABC";但是,这里所做的是创建{ 'A', 'B', 'C', 0 }并将out分配给它;实际上并不修改缓冲区的内容。你应该这样做:
memcpy(out, "ABC", 3);第四个问题:当您从一个String中创建一个byte[]时,您应该指定编码:
new String(c, StandardCharsets.UTF_8);发布于 2014-03-15 11:34:39
我认为在调用bb.position()之前,需要调用bb.get(0)。
https://stackoverflow.com/questions/22423147
复制相似问题