在Python C API中使用多个模块/类型?

时间:2010-01-25 22:57:38

标签: python c binding python-c-api python-extensions

我有两个不同的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甚至无法解析它)。到目前为止,我自己做的扩展一直很轻松,但我很难接受这个扩展。

1 个答案:

答案 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 *