Swig-绑定返回指向导演类指针的引用的方法

时间:2019-03-13 10:36:41

标签: python c++ swig

好吧,标题有点长,但是我想不出一个简短的标题:)所以让我解释一下。

我有一个C ++代码库,其中有一些容器类。这些类具有通过引用或const引用返回项的访问方法。然后在代码库的另一部分中,我分配了object堆的容器,这些容器正在使用导向器功能。

而且我不知道如何专门化object的容器类:对于所有返回对项目的引用(在本例中为对object指针的引用)的容器方法, Swig生成了一个包装代码,由于无效的dy​​namic_cast(基本上是尝试将object **转换为Swig::Director *而导致的包装代码无法编译

我设法用以下代码重现了这个问题。

Test.h

#ifndef TEST_H
#define TEST_H

template< typename T >
class Vector
{

public:

    inline Vector(void)
        : m_Data(nullptr)
        , m_Size(0)
    {
    }

    inline ~Vector(void)
    {
        delete [] m_Data;
    }

    inline void add(const T & item)
    {
        T * data = new T [m_Size + 1];
        for (int i = 0; i < m_Size; ++i)
        {
            data[i] = std::move(m_Data[i]);
        }
        delete [] m_Data;
        m_Data = data;
        m_Data[m_Size++] = item;
    }

    inline const T& item(int index) const
    {
        return m_Data[index];
    }

    inline int count(void) const
    {
        return m_Size;
    }

private:

    T * m_Data;
    int m_Size;

};

class Foo
{

public:

    Foo(void) = default;
    virtual ~Foo(void) = default;

    virtual const char * method(void) const
    {
        return "Foo::method";
    }

};

class Cache
{

public:

    static void add(Foo * item = nullptr)
    {
        m_Cache.add(item == nullptr ? new Foo() : item);
    }

    static const Vector< Foo * > & get(void)
    {
        return m_Cache;
    }

    static Foo * get(int index)
    {
        return m_Cache.item(index);
    }

private:

    static Vector< Foo * > m_Cache;

};

Vector< Foo * > Cache::m_Cache;

#endif // TEST_H

core.i

%module(directors="1") core

// we want to be able to inherit Foo in Python
%feature("director") Foo;

// generate wrappers
%include "Test.h"

// specialize Vector for Foo
%template(FooVector) Vector<Foo*>;

// when compiling the wrapper code, include those
%{
#include "Test.h"
%}

如果您生成Python模块(swig.exe -python -c++ core.i),它可以正常工作,但是生成的core_wrap.cxx文件无法构建,因为为Vector::item生成的包装器代码包含来自{{ 1}}至Foo **

违规行是(结果为Swig::Director *类型)

Foo **

如果我这样手动修复它:

  director = SWIG_DIRECTOR_CAST(result);

然后模块正确编译,一切正常。

所以基本上我的问题是:这是Swig中的错误吗?我做错了吗?是否有解决方法告诉Swig在转换为 director = SWIG_DIRECTOR_CAST(*result); 之前正确解引用我的Foo **结果?

任何帮助表示赞赏:)

1 个答案:

答案 0 :(得分:1)

我找到了解决您问题的方法。

在某些情况下,使用导向器功能和本机RTTI会导致代码无法直接编译,例如从Python使用distutils时。

一种适用于您的示例的解决方案是在编译代码时定义SWIG_DIRECTOR_NORTTI,例如对于GNU编译器,将-DSWIG_DIRECTOR_NORTTI添加到CXXFLAGS