使用 SWIG 在 Python 中包装 C++ 类

Posted

技术标签:

【中文标题】使用 SWIG 在 Python 中包装 C++ 类【英文标题】:Wrapping a C++ class in Python using SWIG 【发布时间】:2011-01-04 17:22:10 【问题描述】:

example.h

#ifndef EXAMPLE_H
#define EXAMPLE_H

class Math 
 public:
    int pi() const;
    void pi(int pi);
 private:
    int _pi;
;

#endif

example.cpp

#include "example.h"

int Math::pi() const 
    return this->_pi;
  
void Math::pi(int pi) 
    this->_pi = pi;

example.swig

%module example
% 
    #define SWIG_FILE_WITH_INIT
    #include "example.h"
%
%include "example.h"

然后我使用以下方法生成包装器“example.py”和“example_wrap.c”:

swig  -python example.swig

当我尝试使用以下方法编译包装类时:

g++ -fPIC -c example.cpp example_wrap.c -I/usr/local/include/python2.6/

我收到以下错误:

example_wrap.cpp: In function "PyObject* Swig_var_Math_get()":
example_wrap.cpp:2725: error: expected primary-expression before "void"
example_wrap.cpp:2725: error: expected ")" before "void"

错误在以下行:

pyobj = SWIG_NewPointerObj(SWIG_as_voidptr(&Math), SWIGTYPE_p_class,  0 );

#define SWIG_as_voidptr(a) (void *)((const void *)(a))

生成包装类“example_wrap.c”是否正确?

【问题讨论】:

你的格式问题很严重。 还不错,他们只是不知道将所有代码缩进四个空格。固定。 【参考方案1】:

我认为swig命令应该是“swig -c++ -python example.swig”

【讨论】:

【参考方案2】:

这里没有足够的信息来确定哪里出了问题,但我有两个想法供您尝试。

    您的g++ 调用正在编译一个C 源文件,就像它是C++ 一样。这保证有效。试试吧

    gcc -I/usr/local/include/python2.6 -fPIC -c example_wrap.c
    gcc -I/usr/local/include/python2.6 -fPIC -c example.cpp
    g++ -shared example_wrap.o example.o -o example.so
    

    (是的,srsly,链接只使用 g++)

    如果这不起作用,请像这样编译example_wrap.c

    gcc -I/usr/local/include/python2.6 -fPIC -c -save-temps example_wrap.c
    

这将以同样的方式失败,但会生成一个名为 example_wrap.i 的文件,这是预处理的结果。这将是巨大的。在该文件中搜索函数Swig_var_Math_get,并将该函数的完整文本 添加到您的问题中(但仅此而已)。

【讨论】:

糟糕,example.cpp 应该使用 g++ 编译。为什么选择 gcc?避免 C++ 修改? 只需要使用g++链接即可。 gcc -c 在应用于 .cpp 源文件时会做正确的事情。但是g++ -c 在应用于.c 源文件时不会 做正确的事情。所以要记住的最简单的规则是,始终使用gcc 进行编译,即使源代码是 C++;仅使用 g++ 链接包含 C++ 的程序。 GCC 支持的任何其他语言都相同:仅使用特定于语言的驱动程序进行链接。就我个人而言,我认为甚至不应该存在特定于语言的驱动程序,但它们确实让您不必确切知道如何调用运行时库。 嗯,是的,你可以使用 gcc 来编译 C++。您也可以使用 gcc 链接 C++,但您必须手动链接 libstdc++ 等。这样一来,我发现将 gcc 用于 C 和 g++ 用于 C++ 更容易。但这只是个人口味。重要的是正确指出了问题和解决方案。 +1【参考方案3】:

感谢您的重播!

-C++ 选项为包装器生成 C++ 类。 swig -c++ -v -python example.swig

我使用 g++ 编译包装器。

g++ -fPIC -c example.cpp example_wrap.cxx -I/usr/local/include/python2.6/

以及以下命令来构建共享对象。当然,我们需要删除多余的包含(-I)和库(-L)。重要的标志是“-shared”和“-fPIC”。

g++ example_wrap.o example.o -L/u01/app/oracle/product/1020.full/lib  -I/usr/local/ssl/include  -L/usr/local/ssl/lib -lclntsh -lssl -lcrypto -ldl -L/usr/local/lib -L/lib64 -L/usr/local/lib/python2.6/ -lboost_system -lboost_filesystem -lboost_thread -lboost_date_time -lglog -lmodpbase64 -lpthread -ldl -lrt -shared -fPIC -o _example.so

【讨论】:

以上是关于使用 SWIG 在 Python 中包装 C++ 类的主要内容,如果未能解决你的问题,请参考以下文章

在 try catch 中包装所有 swig 生成的方法

在 Swig 中通过 ref 返回动态数组

如何在 SWIG 中为 c++ 创建模板函数?

模板包装上的 SWIG 错误

SWIG python初始化一个指向NULL的指针

SWIG:你能否使用 SWIG 专门使用 C++ 头文件使 C++ 在 Python 中可用?