如何使用从C++到java的java.nio.ByteBuffer返回
How to use a java.nio.ByteBuffer return from C++ to Java
这几乎是我如何使Swig正确包装在C中修改为Java Something或其他的char*缓冲区中的问题的重复
如果我使用字节缓冲区而不是Stringbuffer,那么typemap会有什么变化?
我举了一个例子,说明如何使用以下头文件/函数作为测试:
#include <stdio.h>
static void foo(char *buf, int len) {
while(len--)
putchar(*buf++);
}
我的解决方案是修改这个答案,使代理获取ByteBuffer
并将其转换为byte[]
,以便我们传递给JNI代码,然后JNI代码将其转换成指针+长度的组合。
%module test
%{
#include "test.h"
%}
%typemap(jtype) (char *buf, int len) "byte[]"
%typemap(jstype) (char *buf, int len) "java.nio.ByteBuffer"
%typemap(jni) (char *buf, int len) "jbyteArray"
%typemap(javain,pre=" byte[] temp$javainput = new byte[$javainput.capacity()];"
" $javainput.get(temp$javainput);")
(char *buf, int len) "temp$javainput"
%typemap(in,numinputs=1) (char *buf, int len) {
$1 = JCALL2(GetByteArrayElements, jenv, $input, NULL);
$2 = JCALL1(GetArrayLength, jenv, $input);
}
%typemap(freearg) (const signed char *arr, size_t sz) {
// Or use 0 instead of ABORT to keep changes if it was a copy
JCALL3(ReleaseByteArrayElements, jenv, $input, $1, JNI_ABORT);
}
%include "test.h"
这里的新位在javain typemap中,分配一个临时byte[]
,然后使用get
来填充它。实际上,如果您使用的ByteBuffer
支持array()
函数,则应该使用它,即typemap应该只是:
%typemap(javain) (char *buf, int len) "$javainput.array()"
如果您的实现支持它(该方法是可选的,并且可能抛出UnsuportedOperationException
)。
事实上,使用SWIG 2.0可以进一步简化这一点,从前面提到的问题来看,因为我们希望的类型总是为byte
,所以我们可以使用SWIG 2.0中构建的int类型映射来简化我们的接口,现在它变为:
%module test
%{
#include "test.h"
%}
%apply (char *STRING, size_t LENGTH) { (char *buf, int len) }
%typemap(javain) (char *buf, int len) "$javainput.array()"
%typemap(jstype) (char *buf, int len) "java.nio.ByteBuffer"
%include "test.h"
我用以下Java测试了这三个版本:
public class run {
public static void main(String[] argv) {
System.loadLibrary("test");
byte value[] = "hello worldn".getBytes();
java.nio.ByteBuffer buf = java.nio.ByteBuffer.wrap(value);
test.foo(buf);
}
}
为了安全地使用可能不受支持的array()
,您可能想做的是在带有pragma:的函数中添加一个try/catch
%pragma(java) modulecode = %{
private static byte[] buf2bytearr(java.nio.ByteBuffer buf) {
try {
return buf.array();
}
catch (UnsupportedOperationException e) {
byte arr[] = new byte[buf.capacity()];
buf.get(arr);
return arr;
}
}
%}
然后修改类型映射以使用:
%typemap(javain) (char *buf, int len) "buf2bytearr($javainput)"
相关文章:
- 在java中解决这段代码时面临循环中的问题
- 尝试用java代码编译和运行c++代码
- 在这种情况下,java对象是否可以调用本机函数
- 在java中读取c++字节的位字段
- 为什么C++对链表中的下一个节点使用指针,而像 C# 或 Java 这样的语言只使用类 Node 的名称?
- 使用已使用 java 编码的 openssl 解码数据
- SWIG Java 在使用 -DSWIGWORDSIZE64 时将int64_t转换为 jlong
- Android Java USB for native cpp
- 在由Sublime文本3编译后在cmd上显示Java程序输出
- C++ equivalent to Java Map getOrDefault?
- C++ 中的 Java 样式枚举
- 是否有技术原因阻止 Java 中的 final C++ 像 const 一样严格?
- 加密在 Windows、C++ 和 Java 中传输中的数据
- Java从C++回调到C++回调
- 在 c++ 中模拟输入并在 JAVA 中读取它?
- 用C++包装 Java 库 (JNI)
- 使用 TreeSet Java 对反转进行计数
- 验证openssl c++中的签名,这是由JAVA DSA签名的?
- 如何通过 JNI 将 C 字符串表情符号传递给 Java
- 如何使用从C++到java的java.nio.ByteBuffer返回