此问题在某种程度上是此处发布的问题的扩展: 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
答案 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玩游戏,但我认为这不会轻松或值得付出努力。