我有以下三个文件:
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) 中创建类型映射,但我无法正确弄清楚。
先谢谢您的帮助。
%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)
这只会应用类型映射中处理输入参数的部分。