新聞中心
2. 自定義擴展類型:教程
python 允許編寫 C 擴展模塊定義可以從 Python 代碼中操縱的新類型,這很像內(nèi)置的 str 和 list 類型。所有擴展類型的代碼都遵循一個模式,但是在您開始之前,您需要了解一些細節(jié)。這份文件是對這個主題介紹。

成都創(chuàng)新互聯(lián)專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務,包含不限于網(wǎng)站制作、成都網(wǎng)站設計、茌平網(wǎng)絡推廣、微信小程序開發(fā)、茌平網(wǎng)絡營銷、茌平企業(yè)策劃、茌平品牌公關、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運營等,從售前售中售后,我們都將竭誠為您服務,您的肯定,是我們最大的嘉獎;成都創(chuàng)新互聯(lián)為所有大學生創(chuàng)業(yè)者提供茌平建站搭建服務,24小時服務熱線:13518219792,官方網(wǎng)址:www.cdcxhl.com
2.1. 基礎
The CPython runtime sees all Python objects as variables of type PyObject*, which serves as a “base type” for all Python objects. The PyObject structure itself only contains the object’s reference count and a pointer to the object’s “type object”. This is where the action is; the type object determines which (C) functions get called by the interpreter when, for instance, an attribute gets looked up on an object, a method called, or it is multiplied by another object. These C functions are called “type methods”.
所以,如果你想要定義新的擴展類型,需要創(chuàng)建新的類型對象。
這類事情只能用例子解釋,這里用一個最小化但完整的的模塊,定義了新的類型叫做 Custom 在C擴展模塊 custom 里。
備注
這里展示的方法是定義 static 擴展類型的傳統(tǒng)方法??梢赃m合大部分用途。C API也可以定義在堆上分配的擴展類型,使用 PyType_FromSpec() 函數(shù),但不在本入門里討論。
#define PY_SSIZE_T_CLEAN#includetypedef struct {PyObject_HEAD/* Type-specific fields go here. */} CustomObject;static PyTypeObject CustomType = {PyVarObject_HEAD_INIT(NULL, 0).tp_name = "custom.Custom",.tp_doc = PyDoc_STR("Custom objects"),.tp_basicsize = sizeof(CustomObject),.tp_itemsize = 0,.tp_flags = Py_TPFLAGS_DEFAULT,.tp_new = PyType_GenericNew,};static PyModuleDef custommodule = {PyModuleDef_HEAD_INIT,.m_name = "custom",.m_doc = "Example module that creates an extension type.",.m_size = -1,};PyMODINIT_FUNCPyInit_custom(void){PyObject *m;if (PyType_Ready(&CustomType) < 0)return NULL;m = PyModule_Create(&custommodule);if (m == NULL)return NULL;Py_INCREF(&CustomType);if (PyModule_AddObject(m, "Custom", (PyObject *) &CustomType) < 0) {Py_DECREF(&CustomType);Py_DECREF(m);return NULL;}return m;}
這部分很容易理解,這是為了跟上一章能對接上。這個文件定義了三件事:
-
Custom類的對象 object 包含了:CustomObject結(jié)構(gòu),這會為每個Custom實例分配一次。 -
Customtype 的行為:這是CustomType結(jié)構(gòu)體,其定義了一堆標識和函數(shù)指針,會指向解釋器里請求的操作。 -
初始化
custom模塊:PyInit_custom函數(shù)和對應的custommodule結(jié)構(gòu)體。
結(jié)構(gòu)的第一塊是
typedef struct {PyObject_HEAD} CustomObject;
This is what a Custom object will contain. PyObject_HEAD is mandatory at the start of each object struct and defines a field called ob_base of type PyObject, containing a pointer to a type object and a reference count (these can be accessed using the macros Py_TYPE and Py_REFCNT respectively). The reason for the macro is to abstract away the layout and to enable additional fields in debug builds.
備注
注意在宏 PyObject_HEAD 后沒有分號。意外添加分號會導致編譯器提示出錯。
當然,對象除了在 PyObject_HEAD 存儲數(shù)據(jù)外,還有額外數(shù)據(jù);例如,如下定義了標準的Python浮點數(shù):
typedef struct {PyObject_HEADdouble ob_fval;} PyFloatObject;
第二個位是類型對象的定義:
static PyTypeObject CustomType = {PyVarObject_HEAD_INIT(NULL, 0).tp_name = "custom.Custom",.tp_doc = PyDoc_STR("Custom objects"),.tp_basicsize = sizeof(CustomObject),.tp_itemsize = 0,.tp_flags = Py_TPFLAGS_DEFAULT,.tp_new = PyType_GenericNew,};
備注
推薦使用如上C99風格的初始化,以避免列出所有的 PyTypeObject 字段,其中很多是你不需要關心的,這樣也可以避免關注字段的定義順序。
在 object.h 中實際定義的 PyTypeObject 具有比如上定義更多的 字段。 剩余的字段會由 C 編譯器用零來填充,通常的做法是不顯式地指定它們,除非你確實需要它們。
我們先挑選一部分,每次一個字段:
PyVarObject_HEAD_INIT(NULL, 0)
這一行是強制的樣板,用以初始化如上提到的 ob_base 字段:
.tp_name = "custom.Custom",
我們的類型的名稱。 這將出現(xiàn)在我們的對象的默認文本表示形式和某些錯誤消息中,例如:
>>> "" + custom.Custom()Traceback (most recent call last):File "", line 1, in TypeError: can only concatenate str (not "custom.Custom") to str
Note that the name is a dotted name that includes both the module name and the name of the type within the module. The module in this case is custom and the type is Custom, so we set the type name to custom.Custom. Using the real dotted import path is important to make your type compatible with the pydoc and pickle modules.
.tp_basicsize = sizeof(CustomObject),.tp_itemsize = 0,
This is so that Python knows how much memory to allocate when creating new Custom instances. tp_itemsize is only used for variable-sized objects and should otherwise be zero.
備注
If you want your type to be subclassable from Python, and your type has the same tp_basicsize as its base type, you may have problems with multiple inheritance. A Python subclass of your type will have to list your type first in its __bases__, or else it will not be able to call your type’s __new__() method without getting an error. You can avoid this problem by ensuring that your type has a larger value for tp_basicsize than its base type does. Most of the time, this will be true anyway, because either your base type will be object, or else you will be adding data members to your base type, and therefore increasing its size.
我們將類旗標設為 Py_TPFLAGS_DEFAULT。
.tp_flags = Py_TPFLAGS_DEFAULT,
All types should include this constant in their flags. It enables all of the members defined until at least Python 3.3. If you need further members, you will need to OR the corresponding flags.
We provide a doc string for the type in tp_doc.
.tp_doc = PyDoc_STR("Custom objects"),
To enable object creation, we have to provide a tp_new handler. This is the equivalent of the Python method __new__(), but has to be specified explicitly. In this case, we can just use the default implementation provided by the API function PyType_GenericNew().
.tp_new = PyType_GenericNew,
Everything else in the file should be familiar, except for some code in PyInit_custom():
if (PyType_Ready(&CustomType) < 0)return;
This initializes the Custom type, filling in a number of members to the appropriate default values, including ob_type that we initially set to NULL.
Py_INCREF(&CustomType);if (PyModule_AddObject(m, "Custom", (PyObject *) &CustomType) < 0) {Py_DECREF(&CustomType);Py_DECREF(m);return NULL;}
This adds the type to the module dictionary. This allows us to create Custom instances by calling the Custom class:
>>> import custom>>> mycustom = custom.Custom()
That’s it! All that remains is to build it; put the above code in a file called custom.c and:
from distutils.core import setup, Extensionsetup(name="custom", version="1.0",ext_modules=[Extension("custom", ["custom.c"])])
in a file called setup.py; then typing
$ python setup.py build
at a shell should produce a file custom.so in a subdirectory; move to that directory and fire up Python —- you should be able to import custom and play around with Custom objects.
這并不難,對嗎?
Of course, the current Custom type is pretty uninteresting. It has no data and doesn’t do anything. It can’t even be subclassed.
備注
While this documentation showcases the standard distutils module for building C extensions, it is recommended in real-world use cases to use the newer and better-maintained setuptools library. Documentation on how to do this is out of scope for this document and can be found in the Python Packaging User’s Guide.
2.2. Adding data and methods to the Basic example
Let’s extend the basic example to add some data and methods. Let’s also make the type usable as a base class. We’ll create a new module, custom2 that adds these capabilities:
#define PY_SSIZE_T_CLEAN#include#include "structmember.h"typedef struct {PyObject_HEADPyObject *first; /* first name */PyObject *last; /* last name */int number;} CustomObject;static voidCustom_dealloc(CustomObject *self){Py_XDECREF(self->first);Py_XDECREF(self->last);Py_TYPE(self)->tp_free((PyObject *) self);}static PyObject *Custom_new(PyTypeObject *type, PyObject *args, PyObject *kwds){CustomObject *self;self = (CustomObject *) type->tp_alloc(type, 0);if (self != NULL) {self->first = PyUnicode_FromString("");if (self->first == NULL) {Py_DECREF(self);return NULL;}self->last = PyUnicode_FromString("");if (self->last == NULL) {Py_DECREF(self);return NULL;}self->number = 0;}return (PyObject *) self;}static intCustom_init(CustomObject *self, PyObject *args, PyObject *kwds){static char *kwlist[] = {"first", "last", "number", NULL};PyObject *first = NULL, *last = NULL, *tmp;if (!PyArg_ParseTupleAndKeywords(args, kwds, "|OOi", kwlist,&first, &last,&self->number))return -1;if (first) {tmp = self->first;Py_INCREF(first);self->first = first;Py_XDECREF(tmp);}if (last) {tmp = self->last;Py_INCREF(last);self->last = last;Py_XDECREF(tmp);}return 0;}static PyMemberDef Custom_members[] = {{"first", T_OBJECT_EX, offsetof(CustomObject, first), 0,"first name"},{"last", T_OBJECT_EX, offsetof(CustomObject, last), 0,"last name"},{"number", T_INT, offsetof(CustomObject, number), 0,"custom number"},{NULL} /* Sentinel */};static PyObject *Custom_name(CustomObject *self, PyObject *Py_UNUSED(ignored)){if (self->first == NULL) {PyErr_SetString(PyExc_AttributeError, "first");return NULL;}if (self->last == NULL) {PyErr_SetString(PyExc_AttributeError, "last");return NULL;}return PyUnicode_FromFormat("%S %S", self->first, self->last);}static PyMethodDef Custom_methods[] = {{"name", (PyCFunction) Custom_name, METH_NOARGS,"Return the name, combining the first and last name"},{NULL} /* Sentinel */};static PyTypeObject CustomType = {PyVarObject_HEAD_INIT(NULL, 0).tp_name = "custom2.Custom",.tp_doc = PyDoc_STR("Custom objects"),.tp_basicsize = sizeof(CustomObject),.tp_itemsize = 0,.tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE,.tp_new = Custom_new,.tp_init = (initproc) Custom_init,.tp_dealloc = (destructor) Custom_dealloc,.tp_members = Custom_members,.tp_methods = Custom_methods,};static PyModuleDef custommodule = {PyModuleDef_HEAD_INIT,.m_name = "custom2",.m_doc = "Example module that creates an extension type.",.m_size = -1,};PyMODINIT_FUNCPyInit_custom2(void){PyObject *m;if (PyType_Ready(&CustomType) < 0)return NULL;m = PyModule_Create(&custommodule);if (m == NULL)return NULL;Py_INCREF(&CustomType);if (PyModule_AddObject(m, "Custom", (PyObject *) &CustomType) < 0) {Py_DECREF(&CustomType);Py_DECREF(m);return NULL;}return m;}
This version of the module has a number of changes.
We’ve added an extra include:
#include
This include provides declarations that we use to handle attributes, as described a bit later.
The Custom type now has three data attributes in its C struct, first, last, and number. The first and last variables are Python strings containing first and last names. The number attribute is a C integer.
The object structure is updated accordingly:
typedef struct {PyObject_HEADPyObject *first; /* first name */PyObject *last; /* last name */int number;} CustomObject;
Because we now have data to manage, we have to be more careful about object allocation and deallocation. At a minimum, we need a deallocation method:
static voidCustom_dealloc(CustomObject *self){Py_XDECREF(self->first);Py_XDECREF(self->last);Py_TYPE(self)->tp_free((PyObject *) self);}
which is assigned to the tp_dealloc member:
.tp_dealloc = (destructor) Custom_dealloc,
This method first clears the reference counts of the two Python attributes. Py_XDECREF() correctly handles the case where its argument is NULL (which might happen here if tp_new failed midway). It then calls the tp_free member of the object’s type (computed by Py_TYPE(self)) to free the object’s memory. Note that the object’s type might not be CustomType, because the object may be an instance of a subclass.
備注
The explicit cast to destructor above is needed because we defined Custom_dealloc to take a CustomObject * argument, but the tp_dealloc function pointer expects to receive a PyObject * argument. Otherwise, the compiler will emit a warning. This is object-oriented polymorphism, in C!
We want to make sure that the first and last names are initialized to empty strings, so we provide a tp_new implementation:
static PyObject *Custom_new(PyTypeObject *type, PyObject *args, PyObject *kwds){CustomObject *self;self = (CustomObject *) type->tp_alloc(type, 0);if (self != NULL) {self->first = PyUnicode_FromString("");if (self->first == NULL) {Py_DECREF(self);return NULL;}self->last = PyUnicode_FromString("");if (self->last == NULL) {Py_DECREF(self);return NULL;}self->number = 0;}return (PyObject *) self;}
and install it in the tp_new member:
.tp_new = Custom_new,
The tp_new handler is responsible for creating (as opposed to initializing) objects of the type. It is exposed in Python as the __new__() method. It is not required to define a tp_new member, and indeed many extension types will simply reuse PyType_GenericNew() as done in the first version of the Custom type above. In this case, we use the tp_new handler to initialize the first and last attributes to non-NULL default values.
tp_new is passed the type being instantiated (not necessarily CustomType, if a subclass is instantiated) and any arguments passed when the type was called, and is expected to return the instance created. tp_new handlers always accept positional and keyword arguments, but they often ignore the arguments, leaving the argument handling to initializer (a.k.a. tp_init in C or __init__ in Python) methods.
備注
tp_new shouldn’t call tp_init explicitly, as the interpreter will do it itself.
The tp_new implementation calls the tp_alloc slot to allocate memory:
self = (CustomObject *) type->tp_alloc(type, 0);
Since memory allocation may fail, we must check the tp_alloc result against NULL before proceeding.
備注
We didn’t fill the tp_alloc slot ourselves. Rather PyType_Ready() fills it for us by inheriting it from our base class, which is object by default. Most types use the default allocation strategy.
備注
If you are creating a co-operative tp_new (one that calls a base type’s tp_new or __new__()), you must not try to determine what method to call using method resolution order at runtime. Always statically determine what type you are going to call, and call its tp_new directly, or via type->tp_base->tp_new. If you do not do this, Python subclasses of your type that also inherit from other Python-defined classes may not work correctly. (Specifically, you may not be able to create instances of such subclasses without getting a TypeError.)
We also define an initialization function which accepts arguments to provide initial values for our instance:
static intCustom_init(CustomObject *self, PyObject *args, PyObject *kwds){static char *kwlist[] = {"first", "last", "number", NULL};PyObject *first = NULL, *last = NULL, *tmp;if (!PyArg_ParseTupleAndKeywords(args, kwds, "|OOi", kwlist,&first, &last,&self->number))return -1;if (first) {tmp = self->first;Py_INCREF(first);self->first = first;Py_XDECREF(tmp);}if (last) {tmp = self->last;Py_INCREF(last);self->last = last;Py_XDECREF(tmp);}return 0;}
by filling the tp_init slot.
.tp_init = (initproc) Custom_init,
The tp_init slot is exposed in Python as the __init__() method. It is used to initialize an object after it’s created. Initializers always accept positional and keyword arguments, and they should return either 0 on success or -1 on error.
Unlike the tp_new handler, there is no guarantee that tp_init is called at all (for example, the pickle module by default doesn’t call __init__() on unpickled instances). It can also be called multiple times. Anyone can call the __init__() method on our objects. For this reason, we have to be extra careful when assigning the new attribute values. We might be tempted, for example to assign the first member like this:
if (first) {Py_XDECREF(self->first);Py_INCREF(first);self->first = first;}
But this would be risky. Our type doesn’t restrict the type of the first member, so it could be any kind of object. It could have a destructor that causes code to be executed that tries to access the first member; or that destructor could release the Global interpreter Lock and let arbitrary code run in other threads that accesses and modifies our object.
To be paranoid and protect ourselves against this possibility, we almost always reassign members before decrementing their reference counts. When don’t we have to do this?
-
when we absolutely know that the reference count is greater than 1;
-
when we know that deallocation of the object 1 will neither release the GIL nor cause any calls back into our type’s code;
-
when decrementing a reference count in a tp_dealloc handler on a type which doesn’t support cyclic garbage collection 2.
We want to expose our instance variables as attributes. There are a number of ways to do that. The simplest way is to define member definitions:
static PyMemberDef Custom_members[] = {{"first", T_OBJECT_EX, offsetof(CustomObject, first), 0,"first name"},{"last", T_OBJECT_EX, offsetof(CustomObject, last), 0,"last name"},{"number", T_INT, offsetof(CustomObject, number), 0,"custom number"},{NULL} /* Sentinel */};
and put the definitions in the tp_members slot:
.tp_members = Custom_members,
Each member definition has a member name, type, offset, access flags and documentation string. See the 泛型屬性管理 section below for details.
A disadvantage of this approach is that it doesn’t provide a way to restrict the types of objects that can be assigned to the Python attributes. We expect the first and last names to be strings, but any Python objects can be assigned. Further, the attributes can be deleted, setting the C pointers to NULL. Even though we can make sure the members are initialized to non-NULL values, the members can be set to NULL if the attributes are deleted.
We define a single method, Custom.name(), that outputs the objects name as the concatenation of the first and last names.
static PyObject *Custom_name(CustomObject *self, PyObject *Py_UNUSED(ignored)){if (self->first == NULL) {PyErr_SetString(PyExc_AttributeError, "first");return NULL;}if (self->last == NULL) {PyErr_SetString(PyExc_AttributeError, "last");return NULL;}return PyUnicode_FromFormat("%S %S", self->first, self->last);}
The method is implemented as a C function that takes a Custom (or Custom subclass) instance as the first argument. Methods always take an instance as the first argument. Methods often take positional and keyword arguments as well, but in this case we don’t take any and don’t need to accept a positional argument tuple or keyword argument dictionary. This method is equivalent to the Python method:
def name(self):return "%s %s" % (self.first, self.last)
Note that we have to check for the possibility that our first and last members are NULL. This is because they can be deleted, in which case they are set to NULL. It would be better to prevent deletion of these attributes and to restrict the attribute values to be strings. We’ll see how to do that in the next section.
Now that we’ve defined the method, we need to create an array of method definitions:
static PyMethodDef Custom_methods[] = {{"name", (PyCFunction) Custom_name, METH_NOARGS,"Return the name, combining the first and last name"},{NULL} /* Sentinel */};
(note that we used the METH_NOARGS flag to indicate that the method is expecting no arguments other than self)
and assign it to the tp_methods slot:
.tp_methods = Custom_methods,
Finally, we’ll make our type usable as a base class for subclassing. We’ve written our methods carefully so far so that they don’t make any assumptions about the type of the object being created or used, so all we need to do is to add the Py_TPFLAGS_BASETYPE to our class flag definition:
.tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE,
We rename PyInit_custom() to PyInit_custom2(), update the module name in the PyModuleDef struct, and update the full class name in the PyTypeObject struct.
Finally, we update our setup.py file to build the new module:
from distutils.core import setup, Extensionsetup(name="custom", version="1.0",ext_modules=[Extension("custom", ["custom.c"]),Extension("custom2", ["custom2.c"]),])
2.3. Providing finer control over data attributes
In this section, we’ll provide finer control over how the first and last attributes are set in the Custom example. In the previous version of our module, the instance variables first and last could be set to non-string values or even deleted. We want to make sure that these attributes always contain strings.
#define PY_SSIZE_T_CLEAN#include#include "structmember.h"typedef struct {PyObject_HEADPyObject *first; /* first name */PyObject *last; /* last name */int number;} CustomObject;static voidCustom_dealloc(CustomObject *self){Py_XDECREF(self->first);Py_XDECREF(self->last);Py_TYPE(self)->tp_free((PyObject *) self);}static PyObject *Custom_new(PyTypeObject *type, PyObject *args, 網(wǎng)頁標題:創(chuàng)新互聯(lián)Python教程:2.自定義擴展類型:教程
網(wǎng)頁網(wǎng)址:http://www.dlmjj.cn/article/dpiejoc.html


咨詢
建站咨詢
