如何使用从C ++返回给Java的java.nio.ByteBuffer

4
1个回答

0

我已经组合了一个示例,展示如何使用以下头文件/函数进行测试:

#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中的内置类型映射来简化我们的接口,现在变成:
%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 world\n".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)"

Flexo - 非常感谢您 - 我有一个带有四个参数的函数 - 我为此创建了接口定义,但遇到了麻烦。我附上了我的接口,当然在另一个问题中它无法工作,这应该让您了解我正在做什么。 - Yogesh Devi

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接