让 SWIG 理解 char** 以便在 Java 中使用它

Sib*_*ibi 2 c++ java swig

我有以下三个文件:

sab1.hpp:

class CRectangle 

{
    int x, y;

public:
    void set_values (int,int);
    void print(char **db);
    int area ();
};
Run Code Online (Sandbox Code Playgroud)

sab1.cpp:

#include <iostream>
#include "sab1.hpp"
using namespace std;

int CRectangle::area () 
{
    return (x*y);
}

void CRectangle::set_values (int a, int b) 
{
    x = a;
    y = b;
}

void CRectangle::print(char **db) 
{
    cout <<db[0];
}

int main () 
{
    CRectangle rect;
    char *a[] = {"Hi", "bye" }  ;

    rect.set_values (3,4);
    cout << "area: " << rect.area();
    rect.print(&a[0]);
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

sab1.i:

%module Rec
%{
/* Includes the header in the wrapper code */
#include "sab1.hpp"
%}
class CRectangle {
    int x, y;
public:
    void set_values (int,int);
    void print(char **db);
    int area ();
};
Run Code Online (Sandbox Code Playgroud)

我使用以下常用命令创建各种中间文件:

1) 痛饮-c++-java sab1.i

2) g++ -fpic -c sab1.cpp sab1_wrap.cxx -I /usr/lib/jvm/java-6-openjdk-i386/include/ -I /usr/lib/jvm/java-6-openjdk-i386/include /linux/

然后我使用以下命令创建一个共享库: g++ --shared sab1.o sab1_wrap.o -o libegs.so

然后我创建一个以下 Java 文件来访问 C++ 类的函数:

public class Test 
{
    public static void main(String args[]) 
    {
        System.loadLibrary("egs");
        CRectangle a = new CRectangle();
        a.set_values(4,5);
        System.out.println(a.area());
        a.print("Hi");
    }
}
Run Code Online (Sandbox Code Playgroud)

当我编译 Java 文件 (javac -classpath . Test.java ) 时,出现以下错误:

Test.java:17: print(SWIGTYPE_p_p_char) in CRectangle cannot be applied to (java.lang.String)  a.print("Hi");
 ^
Run Code Online (Sandbox Code Playgroud)

那么,怎样才能让Java代码理解char **变量呢?我尝试在 SWIG 接口文件 (sab1.i) 中创建类型映射,但我无法正确弄清楚。

先谢谢您的帮助。

Okt*_*ist 5

%module Rec
%include "various.i"

%apply char **STRING_ARRAY { char **db }

%include "sab1.hpp"
Run Code Online (Sandbox Code Playgroud)

您链接到的文件various.i应该以这种方式工作,只需将其放在 SWIG 可以找到它的地方,在您的项目包含目录或 SWIG 包含路径中。

使用%include代替#include "sab1.hpp"意味着我们不必复制我们想要包装的类的定义。该%apply指令将类型映射 from 应用于具有 name 类型various.i的所有参数。如果您想要包装其他参数名称,请像这样添加它们:char**db

%apply char **STRING_ARRAY { char **db, char **foo, char **bar }
Run Code Online (Sandbox Code Playgroud)

如果您只想包装所有类型的参数char**(不安全!),只需省略名称即可:

%apply char **STRING_ARRAY { char ** }
Run Code Online (Sandbox Code Playgroud)

由于您的 C++ 代码期望数组包含 2 个字符串,因此只要您String从 Java 传递一个包含 2 个或更多字符串的数组,就应该没问题。类型映射将向数组添加第三个空指针,但您的 C++ 代码不会注意到。如果您传递一个少于 2String秒的数组,您的 C++ 代码可能会崩溃。

如果您包装的任何 C++ 函数也返回字符串数组,那么您就有麻烦了。SWIG 将尝试将它们转换回 Java 数组,但当它在数组中查找第三个元素时可能会崩溃(它将继续查找元素,直到找到空指针)。如果你不介意丢弃返回值,你可以这样做:

%module Rec
%include "various.i"

%typemap(jni)     char **db = char **STRING_ARRAY;
%typemap(jtype)   char **db = char **STRING_ARRAY;
%typemap(jstype)  char **db = char **STRING_ARRAY;
%typemap(in)      char **db = char **STRING_ARRAY;
%typemap(freearg) char **db = char **STRING_ARRAY;
%typemap(javain)  char **db = char **STRING_ARRAY;

%include "sab1.hpp"
Run Code Online (Sandbox Code Playgroud)

这只会应用类型映射中处理输入参数的部分。