我有两个不同的Python扩展模块;让我们称它们为A和B.模块A包含一个称为容器的存储类类型,我想在模块B中使用它作为类方法的返回类型。
我似乎无法找到关于我应该怎么做的任何文档。我大致按照本文创建模块/类,除了我没有将所有方法声明为静态,因此可以访问它们:http://nedbatchelder.com/text/whirlext.html
我的问题是,如何创建一个容器实例,我可以将其传递回模块B中类方法的PyObject *返回值?容器定义如下所示:
typedef struct {
PyObject_HEAD
storageclass* cnt_;
} container;
我尝试在相关方法中执行以下操作,其中container_init是我已注册为容器类的tp_init的方法:
pycnt::container* retval;
pycnt::container_init(retval, NULL, NULL);
return (PyObject*)retval;
然而,根据Python解释器,我回到了我调用该方法的类。 (即myclassinstance.mymethod()正在返回myclassinstance)。
我显然是以错误的方式解决这个问题,但我不知道正确的方法是什么。有什么建议?只是为了削减任何人建议它 - 不,我对使用SWIG或Boost :: Python不感兴趣。我已经尝试过了,容器的底层存储类也没有很好地兼容(SWIG甚至无法解析它)。到目前为止,我自己做的扩展一直很轻松,但我很难接受这个扩展。
答案 0 :(得分:3)
你的问题是type->tp_init
没有做你想做的事。在分配对象以进行实例化之后调用type->tp_init
。您首先必须使用type->tp_new
分配对象,然后将该对象作为第一个参数传递给type->tp_init
。由于您将未初始化的指针传递给tp_init
,因此所有投注均已关闭。但是,您通常不直接调用任一函数,而是在类型对象本身上调用PyObject_Call*()
之一来创建新实例。或者提供一个普通的C函数来创建一个新的实例,即la PyFoo_New()
。
也就是说,在两个扩展模块之间进行相互通信的常用方法是通过Python API来实现。如果您希望模块B导入并使用模块A,则调用PyImport_Import()
以获取模块对象,PyObject_GetAttrString()
以获取您关注的类型对象,并调用PyObject_Call*()
之一来实例化它。您想要存储指向该类型的指针的任何地方,您只需要PyObject *
。