問題:SWIGを使用してPythonでいくつかのc++コードをラップしました。Python側では、ラップされたc ++ポインターを取得し、それをサブクラスへのポインターとしてダウンキャストしたいと思います。このダウンキャストを行う新しいc++関数をSWIG.iファイルに追加しましたが、Pythonから呼び出すと、TypeErrorが発生します。
詳細は次のとおりです。
BaseとDerivedの2つのc++クラスがあります。DerivedはBaseのサブクラスです。3番目のクラスであるContainerがあります。これには、Derivedが含まれており、それにアクセサーを提供します。アクセサは、次のようにDerivedをconst Base&として返します。
class Container {
public:
const Base& GetBase() const {
return derived_;
}
private:
Derived derived_;
};
SWIGを使用してこれらのクラスをPythonでラップしました。私のPythonコードでは、Base参照をDerivedにダウンキャストしたいと思います。これを行うために、ダウンキャストを行うc++のヘルパー関数をswig.iファイルに書き込みました。
%inline %{
Derived* CastToDerived(Base* base) {
return static_cast<Derived*>(base);
}
%}
私のPythonコードでは、次のダウンキャスト関数を呼び出します。
base = container.GetBase()
derived = CastToDerived(base)
これを行うと、次のエラーが発生します。
TypeError: in method 'CastToDerived', argument 1 of type 'Base *'
なぜこれが起こっているのでしょうか?
参考までに、SWIGによって生成された.cxxファイルの関連ビットを次に示します。つまり、元の関数とそのpython-interface-ified doppelganger:
Derived* CastToDerived(Base* base) {
return static_cast<Derived*>(base);
}
// (lots of other generated code omitted)
SWIGINTERN PyObject *_wrap_CastToDerived(PyObject *SWIGUNUSEDPARM(self), PyObject *args) {
PyObject *resultobj = 0;
Base *arg1 = (Base *) 0 ;
void *argp1 = 0 ;
int res1 = 0 ;
PyObject * obj0 = 0 ;
Derived *result = 0 ;
if (!PyArg_ParseTuple(args,(char *)"O:CastToDerived",&obj0)) SWIG_fail;
res1 = SWIG_ConvertPtr(obj0, &argp1,SWIGTYPE_p_Base, 0 | 0 );
if (!SWIG_IsOK(res1)) {
SWIG_exception_fail(SWIG_ArgError(res1), "in method '" "CastToDerived" "', argument " "1"" of type '" "Base *""'");
}
arg1 = reinterpret_cast< Base * >(argp1);
result = (Derived *)CastToDerived(arg1);
resultobj = SWIG_NewPointerObj(SWIG_as_voidptr(result), SWIGTYPE_p_Derived, 0 | 0 );
return resultobj;
fail:
return NULL;
}
どんな助けでも大歓迎です。
-マット