4

これは、C で Java 何かまたはその他として変更された char* バッファーを Swig で正しくラップするにはどうすればよいですか?での質問のほとんどの繰り返しです。...
bytebuffer を使用する場合、 Stringbuffer の代わりに、 typemap の変更点は何ですか?

4

1 に答える 1

0

次のヘッダーファイル/関数をテストとして使用して、これを行う方法の例をまとめました。

#include <stdio.h>

static void foo(char *buf, int len) {
  while(len--)
    putchar(*buf++);
}

私の解決策は、プロキシがaを取得し、それをaに変換してJNIコードに渡すようにこの回答を変更し、JNIコードがポインターと長さの組み合わせに変換することでした。ByteBufferbyte[]

%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タイプマップにあり、一時的なものを割り当ててから、それを埋めるためにbyte[]使用します。get実際には、サポートを使用している場合は代わりに使用する必要があるarray()関数があります。つまり、タイプマップは次のようになります。ByteBuffer

%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"

私はこれの3つのバージョンすべてを次の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()に行うには、プラグマを使用して関数に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)"
于 2012-07-24T10:22:07.797 に答える