Swig shared_ptr宏与模板类和派生类

时间:2012-07-08 03:38:50

标签: c++ python swig

此问题在某种程度上是此处发布的问题的扩展: SWIG_SHARED_PTR macro with templated class 虽然问题可能完全不相关。

基本设置如下:我试图让SWIG将模板化的类包装为shared_ptr。所以接口文件看起来应该是这样的

%shared_ptr(template_instance)
%include template_class.cpp
%template(vector_instance) template_class<int>;

现在的问题是template_class有很多派生类,这会在swig中引发很多警告,然后构建错误。这些类不需要作为shared_ptr来处理,所以我宁愿忽略上面代码生成的警告。错误的解决方案似乎是:

%shared_ptr(template_derived1)
%shared_ptr(template_derived2)
.
.
.
%shared_ptr(template_derivedn)
%shared_ptr(template_instance)
%include template_class.cpp
%template(vector_instance) template_class<int>;

这是有效的,但是这是一个巨大的混乱,我认为将所有内容表示为shared_ptr(它是什么?)必定会有一些缺点。这周围有人吗?

编辑:使用特定示例进行更新

test.h

class Base
{
  int base_member;
};

class Derived : public Base
{
  int derived_member;
};

test.i

%module test
%{
#include "test.h"
#include <boost/shared_ptr.hpp>
  %}

%include <boost_shared_ptr.i>
%shared_ptr(Base)
%include test.h

命令:

swig -python -c++ test.i 
g++ -fPIC -I /usr/include/python2.7 -c test_wrap.cxx

在这个剥离的示例中,swig调用发出警告,g ++调用给出错误。请注意,我删除了模板,因为它似乎不是问题的一个组成部分。

通过注释

解决错误
%shared_ptr(Base)

swig生成的警告是:

test.h:10: Warning 520: Derived class 'Derived' of 'Base' is not similarly marked as a smart pointer

来自g ++的错误是:

test_wrap.cxx: In function ‘PyObject* _wrap_delete_Derived(PyObject*, PyObject*)’:
test_wrap.cxx:3155:22: error: ‘smartarg1’ was not declared in this scope

1 个答案:

答案 0 :(得分:4)

这里的警告是因为你需要告诉SWIG关于整个类层次结构,而不仅仅是基类,以便能够有效地使用智能指针。它需要能够在智能指针之间进行转换,以便任何带有Base智能指针的内容也可以接受Derived。所以你的界面文件必须是:

%module test
%{
#include "test.h"
#include <boost/shared_ptr.hpp>
%}

%include <boost_shared_ptr.i>
%shared_ptr(Base)
%shared_ptr(Derived)
%include "test.h"

解决了被警告的问题并生成了在我的机器上编译好的代码。

如果您不想告诉SWIG所有派生类型,最简单的方法是从SWIG中完全隐藏类型 - 只从您要包装的内容中公开Base类型。您可以通过几种方式实现这一点,最不干扰的是将接口文件更改为:

%module test
%{
#include "test.h"
#include <boost/shared_ptr.hpp>
%}

%include <boost_shared_ptr.i>
%ignore Derived;
%shared_ptr(Base)
%include "test.h"

这只会导致Base被包装,因此无法编译无法编译的代码。

或者,因为每个类型仍然需要%ignore,您可以修改头文件以完全隐藏SWIG中Derived的声明/定义:

class Base
{
  int base_member;
};
#ifndef SWIG
class Derived : public Base
{
  int derived_member;
};
#endif

如果你的项目是有组织的,每个类型(粗略地)有一个头文件,你可以通过简单地不使用%include和基本文件以外的文件来更简单地执行此操作。

如果你仍然想要包装它们,但不是作为smart_ptr,我认为你将不得不接受将会有很多%smart_ptr - 你可以自动生成它们吗?您可以使用modules玩游戏,但我认为这不会轻松或值得付出努力。