Python C-API使len(…)与扩展类一起工作
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了Python C-API使len(…)与扩展类一起工作,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含2877字,纯文字阅读大概需要5分钟。
内容图文
![Python C-API使len(…)与扩展类一起工作](/upload/InfoBanner/zyjiaocheng/812/a681688778d148d990dd97613ded557c.jpg)
在Python中创建类时,我可以简单地创建一个def __len __(self):方法来使len(InstanceOfMyClass)工作,但是我无法通过C-API找到如何使用扩展类来完成此操作.
我尝试添加__len__方法,但似乎无效
{"__len__",(PyCFunction)&TestClass_GetLen,METH_NOARGS,""},
Python测试代码:
def test(my_instance):
x = len(my_instance)#exception
return x
TypeError: object of type 'test_module.test_class' has no len()
TestClass的代码
struct TestClass;
static int TestClass_Init(TestClass *self, PyObject *args, PyObject* kwds);
static void TestClass_Dealloc(TestClass *self);
static PyObject* TestClass_GetLen(TestClass *self);
struct TestClass
{
PyObject_HEAD;
};
static PyMethodDef TestClass_methods[] =
{
{"__len__",(PyCFunction)&TestClass_GetLen,METH_O,""},
{NULL}
};
static PyTypeObject TestClass_type = {PyObject_HEAD_INIT(NULL)};
bool InitTestClass(PyObject *module)
{
TestClass_type.tp_basicsize = sizeof(TestClass);
TestClass_type.tp_name = PY_MODULE_NAME".TestClass";
TestClass_type.tp_doc = "";
TestClass_type.tp_flags = Py_TPFLAGS_DEFAULT;
TestClass_type.tp_methods = TestClass_methods;
TestClass_type.tp_new = PyType_GenericNew;
TestClass_type.tp_init = (initproc)TestClass_Init;
TestClass_type.tp_dealloc = (destructor)TestClass_Dealloc;
if(PyType_Ready(TestClass_type) < 0) return false;
Py_INCREF(TestClass_type);
PyModule_AddObject(module, "TestClass", (PyObject*)&TestClass_type);
return true;
};
void TestClass_Dealloc(TestClass *self)
{
Py_TYPE(self)->tp_free((PyObject*)self);
}
int TestClass_Init(TestClass *self, PyObject *args, PyObject* kwds)
{
return 0;
}
PyObject* TestClass_GetLen(TestClass *self)
{
return PyLong_FromLong(55);
}
解决方法:
正如Igniacio所说,正确的方法是填充类型对象的tp_as_sequence成员.这是一个最小的例子:
#include <Python.h>
/**
* C structure and methods definitions
*/
typedef struct {
PyObject_HEAD;
} TestObject;
static Py_ssize_t
TestClass_len(TestObject* self)
{
return 55;
}
/**
* Python definitions
*/
static PySequenceMethods TestClass_sequence_methods = {
TestClass_len, /* sq_length */
};
static PyTypeObject TestClass = {
PyObject_HEAD_INIT(NULL)
0, /*ob_size*/
"testmodule.TestClass", /*tp_name*/
sizeof(TestObject), /*tp_basicsize*/
};
/**
* Module entry point
*/
#ifndef PyMODINIT_FUNC /* declarations for DLL import/export */
#define PyMODINIT_FUNC void
#endif
PyMODINIT_FUNC
inittestmodule(void)
{
PyObject* m;
TestClass.tp_new = PyType_GenericNew;
TestClass.tp_as_sequence = &TestClass_sequence_methods;
TestClass.tp_flags = Py_TPFLAGS_DEFAULT;
if (PyType_Ready(&TestClass) < 0)
return;
m = Py_InitModule3("testmodule", NULL, "");
Py_INCREF(&TestClass);
PyModule_AddObject(m, "TestClass", (PyObject *)&TestClass);
}
另一种(更复杂和更慢)的方法是使用PyCFunction_New,PyMethod_New和PyObject_SetAttr在模块的init函数中动态地向类添加__len__方法.这看起来更像你粘贴的代码,除了你在C方法表中定义__len__,这是行不通的.
内容总结
以上是互联网集市为您收集整理的Python C-API使len(…)与扩展类一起工作全部内容,希望文章能够帮你解决Python C-API使len(…)与扩展类一起工作所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。