Хорошо, название немного длинное, но я не могу найти более короткое :) Итак, позвольте мне объяснить.
У меня есть кодовая база C ++, где у нас есть несколько контейнерных классов. Эти классы имеют методы доступа, возвращающие элементы по ссылке или константной ссылке. Затем в другой части кодовой базы у меня выделены контейнеры с кучей object
, которые используют функцию директора.
И я не могу понять, как специализировать наши контейнерные классы для object
: для всех методов контейнера, возвращающих ссылки на элементы (в данном случае ссылка на указатель object
), Swig генерирует оболочку код, который не компилируется из-за недопустимого dynamic_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
содержит недопустимый dynamic_cast от Foo **
до Swig::Director *
Неправильная строка (где результат типа Foo **
)
director = SWIG_DIRECTOR_CAST(result);
И если я вручную исправлю это так:
director = SWIG_DIRECTOR_CAST(*result);
Затем модуль правильно компилируется, и все работает нормально.
Так что в основном мой вопрос: это ошибка в Swig? Я делаю что-то неправильно? Есть ли обходной путь, чтобы сказать Swig правильно разыменовать мой Foo **
результат перед приведением к Swig::Director *
?
Любая помощь приветствуется:)