类型对象
********

Python 对象系统中最重要的一个结构体也许是定义新类型的结构体:
"PyTypeObject" 结构体。 类型对象可以使用任何 "PyObject_*" 或
"PyType_*" 函数来处理，但并未提供大多数 Python 应用程序会感兴趣的东西
。 这些对象是对象行为的基础，所以它们对解释器本身及任何实现新类型的扩
展模块都非常重要。

与大多数标准类型相比，类型对象相当大。这么大的原因是每个类型对象存储了
大量的值，大部分是C函数指针，每个指针实现了类型功能的一小部分。本节将
详细描述类型对象的字段。这些字段将按照它们在结构中出现的顺序进行描述。

除了下面的快速参考， 例子 小节提供了快速了解 "PyTypeObject" 的含义和用
法的例子。


快速参考
========


"tp_方法槽"
-----------

+--------------------+--------------------+--------------------+----+----+----+----+
| PyTypeObject 槽    | Type               | 特殊方法/属性      | 信息 [2]          |
| [1]                |                    |                    |                   |
|                    |                    |                    +----+----+----+----+
|                    |                    |                    | O  | T  | D  | I  |
|                    |                    |                    |    |    |    |    |
|====================|====================|====================|====|====|====|====|
| <R> "tp_name"      | const char *       | __name__           | X  | X  |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_basicsize"     | "Py_ssize_t"       |                    | X  | X  |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_itemsize"      | "Py_ssize_t"       |                    |    | X  |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_dealloc"       | "destructor"       |                    | X  | X  |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_vectorcall_of  | "Py_ssize_t"       |                    |    | X  |    | X  |
| fset"              |                    |                    |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| ("tp_getattr")     | "getattrfunc"      | __getattribute__,  |    |    |    | G  |
|                    |                    | __getattr__        |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| ("tp_setattr")     | "setattrfunc"      | __setattr__,       |    |    |    | G  |
|                    |                    | __delattr__        |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_as_async"      | "PyAsyncMethods" * | 子方法槽（方法域） |    |    |    | %  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_repr"          | "reprfunc"         | __repr__           | X  | X  |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_as_number"     | "PyNumberMethods"  | 子方法槽（方法域） |    |    |    | %  |
|                    | *                  |                    |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_as_sequence"   | "PySequenceMethod  | 子方法槽（方法域） |    |    |    | %  |
|                    | s" *               |                    |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_as_mapping"    | "PyMappingMethods" | 子方法槽（方法域） |    |    |    | %  |
|                    | *                  |                    |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_hash"          | "hashfunc"         | __hash__           | X  |    |    | G  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_call"          | "ternaryfunc"      | __call__           |    | X  |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_str"           | "reprfunc"         | __str__            | X  |    |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_getattro"      | "getattrofunc"     | __getattribute__,  | X  | X  |    | G  |
|                    |                    | __getattr__        |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_setattro"      | "setattrofunc"     | __setattr__,       | X  | X  |    | G  |
|                    |                    | __delattr__        |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_as_buffer"     | "PyBufferProcs" *  |                    |    |    |    | %  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_flags"         | unsigned long      |                    | X  | X  |    | ?  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_doc"           | const char *       | __doc__            | X  | X  |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_traverse"      | "traverseproc"     |                    |    | X  |    | G  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_clear"         | "inquiry"          |                    |    | X  |    | G  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_richcompare"   | "richcmpfunc"      | __lt__, __le__,    | X  |    |    | G  |
|                    |                    | __eq__, __ne__,    |    |    |    |    |
|                    |                    | __gt__, __ge__     |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| ("tp_weaklistoffs  | "Py_ssize_t"       |                    |    | X  |    | ?  |
| et")               |                    |                    |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_iter"          | "getiterfunc"      | __iter__           |    |    |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_iternext"      | "iternextfunc"     | __next__           |    |    |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_methods"       | "PyMethodDef" []   |                    | X  | X  |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_members"       | "PyMemberDef" []   |                    |    | X  |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_getset"        | "PyGetSetDef" []   |                    | X  | X  |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_base"          | "PyTypeObject" *   | __base__           |    |    | X  |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_dict"          | "PyObject" *       | __dict__           |    |    | ?  |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_descr_get"     | "descrgetfunc"     | __get__            |    |    |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_descr_set"     | "descrsetfunc"     | __set__,           |    |    |    | X  |
|                    |                    | __delete__         |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| ("tp_dictoffset")  | "Py_ssize_t"       |                    |    | X  |    | ?  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_init"          | "initproc"         | __init__           | X  | X  |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_alloc"         | "allocfunc"        |                    | X  |    | ?  | ?  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_new"           | "newfunc"          | __new__            | X  | X  | ?  | ?  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_free"          | "freefunc"         |                    | X  | X  | ?  | ?  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_is_gc"         | "inquiry"          |                    |    | X  |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| <"tp_bases">       | "PyObject" *       | __bases__          |    |    | ~  |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| <"tp_mro">         | "PyObject" *       | __mro__            |    |    | ~  |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| ["tp_cache"]       | "PyObject" *       |                    |    |    |         |
+--------------------+--------------------+--------------------+----+----+----+----+
| ["tp_subclasses"]  | void *             | __subclasses__     |    |    |         |
+--------------------+--------------------+--------------------+----+----+----+----+
| ["tp_weaklist"]    | "PyObject" *       |                    |    |    |         |
+--------------------+--------------------+--------------------+----+----+----+----+
| ("tp_del")         | "destructor"       |                    |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| ["tp_version_tag"] | unsigned int       |                    |    |    |         |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_finalize"      | "destructor"       | __del__            |    |    |    | X  |
+--------------------+--------------------+--------------------+----+----+----+----+
| "tp_vectorcall"    | "vectorcallfunc"   |                    |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+
| ["tp_watched"]     | unsigned char      |                    |    |    |    |    |
+--------------------+--------------------+--------------------+----+----+----+----+

[1] **（）**：括号中的插槽名称表示（实际上）已弃用。

    **<>**: 尖括号内的名称在初始时应设为 "NULL" 并被视为是只读的。

    **[]**: 方括号内的名称仅供内部使用。

    **<R>** (作为前缀) 表示字段是必需的 (不能是 "NULL")。

[2] 列：

    **"O"**:  在 "PyBaseObject_Type" 上设置

    **"T"**:  在 "PyType_Type" 上设置

    **"D"**: 默认设置(如果方法槽被设置为NULL)

       X - PyType_Ready sets this value if it is NULL
       ~ - PyType_Ready always sets this value (it should be NULL)
       ? - PyType_Ready may set this value depending on other slots

       Also see the inheritance column ("I").

    **"I"**: 继承

       X - type slot is inherited via *PyType_Ready* if defined with a *NULL* value
       % - the slots of the sub-struct are inherited individually
       G - inherited, but only in combination with other slots; see the slot's description
       ? - it's complicated; see the slot's description

    注意，有些方法槽是通过普通属性查找链有效继承的。


子方法槽（方法域）
------------------

+----------------------------+-------------------+--------------+
| 方法槽                     | Type              | 特殊方法     |
|============================|===================|==============|
| "am_await"                 | "unaryfunc"       | __await__    |
+----------------------------+-------------------+--------------+
| "am_aiter"                 | "unaryfunc"       | __aiter__    |
+----------------------------+-------------------+--------------+
| "am_anext"                 | "unaryfunc"       | __anext__    |
+----------------------------+-------------------+--------------+
| "am_send"                  | "sendfunc"        |              |
+----------------------------+-------------------+--------------+
|                                                               |
+----------------------------+-------------------+--------------+
| "nb_add"                   | "binaryfunc"      | __add__      |
|                            |                   | __radd__     |
+----------------------------+-------------------+--------------+
| "nb_inplace_add"           | "binaryfunc"      | __iadd__     |
+----------------------------+-------------------+--------------+
| "nb_subtract"              | "binaryfunc"      | __sub__      |
|                            |                   | __rsub__     |
+----------------------------+-------------------+--------------+
| "nb_inplace_subtract"      | "binaryfunc"      | __isub__     |
+----------------------------+-------------------+--------------+
| "nb_multiply"              | "binaryfunc"      | __mul__      |
|                            |                   | __rmul__     |
+----------------------------+-------------------+--------------+
| "nb_inplace_multiply"      | "binaryfunc"      | __imul__     |
+----------------------------+-------------------+--------------+
| "nb_remainder"             | "binaryfunc"      | __mod__      |
|                            |                   | __rmod__     |
+----------------------------+-------------------+--------------+
| "nb_inplace_remainder"     | "binaryfunc"      | __imod__     |
+----------------------------+-------------------+--------------+
| "nb_divmod"                | "binaryfunc"      | __divmod__   |
|                            |                   | __rdivmod__  |
+----------------------------+-------------------+--------------+
| "nb_power"                 | "ternaryfunc"     | __pow__      |
|                            |                   | __rpow__     |
+----------------------------+-------------------+--------------+
| "nb_inplace_power"         | "ternaryfunc"     | __ipow__     |
+----------------------------+-------------------+--------------+
| "nb_negative"              | "unaryfunc"       | __neg__      |
+----------------------------+-------------------+--------------+
| "nb_positive"              | "unaryfunc"       | __pos__      |
+----------------------------+-------------------+--------------+
| "nb_absolute"              | "unaryfunc"       | __abs__      |
+----------------------------+-------------------+--------------+
| "nb_bool"                  | "inquiry"         | __bool__     |
+----------------------------+-------------------+--------------+
| "nb_invert"                | "unaryfunc"       | __invert__   |
+----------------------------+-------------------+--------------+
| "nb_lshift"                | "binaryfunc"      | __lshift__   |
|                            |                   | __rlshift__  |
+----------------------------+-------------------+--------------+
| "nb_inplace_lshift"        | "binaryfunc"      | __ilshift__  |
+----------------------------+-------------------+--------------+
| "nb_rshift"                | "binaryfunc"      | __rshift__   |
|                            |                   | __rrshift__  |
+----------------------------+-------------------+--------------+
| "nb_inplace_rshift"        | "binaryfunc"      | __irshift__  |
+----------------------------+-------------------+--------------+
| "nb_and"                   | "binaryfunc"      | __and__      |
|                            |                   | __rand__     |
+----------------------------+-------------------+--------------+
| "nb_inplace_and"           | "binaryfunc"      | __iand__     |
+----------------------------+-------------------+--------------+
| "nb_xor"                   | "binaryfunc"      | __xor__      |
|                            |                   | __rxor__     |
+----------------------------+-------------------+--------------+
| "nb_inplace_xor"           | "binaryfunc"      | __ixor__     |
+----------------------------+-------------------+--------------+
| "nb_or"                    | "binaryfunc"      | __or__       |
|                            |                   | __ror__      |
+----------------------------+-------------------+--------------+
| "nb_inplace_or"            | "binaryfunc"      | __ior__      |
+----------------------------+-------------------+--------------+
| "nb_int"                   | "unaryfunc"       | __int__      |
+----------------------------+-------------------+--------------+
| "nb_reserved"              | void *            |              |
+----------------------------+-------------------+--------------+
| "nb_float"                 | "unaryfunc"       | __float__    |
+----------------------------+-------------------+--------------+
| "nb_floor_divide"          | "binaryfunc"      | __floordiv__ |
+----------------------------+-------------------+--------------+
| "nb_inplace_floor_divide"  | "binaryfunc"      | __ifloordiv  |
|                            |                   | __           |
+----------------------------+-------------------+--------------+
| "nb_true_divide"           | "binaryfunc"      | __truediv__  |
+----------------------------+-------------------+--------------+
| "nb_inplace_true_divide"   | "binaryfunc"      | __itruediv__ |
+----------------------------+-------------------+--------------+
| "nb_index"                 | "unaryfunc"       | __index__    |
+----------------------------+-------------------+--------------+
| "nb_matrix_multiply"       | "binaryfunc"      | __matmul__   |
|                            |                   | __rmatmul__  |
+----------------------------+-------------------+--------------+
| "nb_inplace_matrix_multip  | "binaryfunc"      | __imatmul__  |
| ly"                        |                   |              |
+----------------------------+-------------------+--------------+
|                                                               |
+----------------------------+-------------------+--------------+
| "mp_length"                | "lenfunc"         | __len__      |
+----------------------------+-------------------+--------------+
| "mp_subscript"             | "binaryfunc"      | __getitem__  |
+----------------------------+-------------------+--------------+
| "mp_ass_subscript"         | "objobjargproc"   | __setitem__, |
|                            |                   | __delitem__  |
+----------------------------+-------------------+--------------+
|                                                               |
+----------------------------+-------------------+--------------+
| "sq_length"                | "lenfunc"         | __len__      |
+----------------------------+-------------------+--------------+
| "sq_concat"                | "binaryfunc"      | __add__      |
+----------------------------+-------------------+--------------+
| "sq_repeat"                | "ssizeargfunc"    | __mul__      |
+----------------------------+-------------------+--------------+
| "sq_item"                  | "ssizeargfunc"    | __getitem__  |
+----------------------------+-------------------+--------------+
| "sq_ass_item"              | "ssizeobjargproc" | __setitem__  |
|                            |                   | __delitem__  |
+----------------------------+-------------------+--------------+
| "sq_contains"              | "objobjproc"      | __contains__ |
+----------------------------+-------------------+--------------+
| "sq_inplace_concat"        | "binaryfunc"      | __iadd__     |
+----------------------------+-------------------+--------------+
| "sq_inplace_repeat"        | "ssizeargfunc"    | __imul__     |
+----------------------------+-------------------+--------------+
|                                                               |
+----------------------------+-------------------+--------------+
| "bf_getbuffer"             | "getbufferproc()" |              |
+----------------------------+-------------------+--------------+
| "bf_releasebuffer"         | "releasebufferpr  |              |
|                            | oc()"             |              |
+----------------------------+-------------------+--------------+


槽位 typedef
------------

+-------------------------------+-------------------------------+------------------------+
| typedef                       | 参数类型                      | 返回类型               |
|===============================|===============================|========================|
| "allocfunc"                   | "PyTypeObject" * "Py_ssize_t" | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "destructor"                  | "PyObject" *                  | void                   |
+-------------------------------+-------------------------------+------------------------+
| "freefunc"                    | void *                        | void                   |
+-------------------------------+-------------------------------+------------------------+
| "traverseproc"                | "PyObject" * "visitproc" void | int                    |
|                               | *                             |                        |
+-------------------------------+-------------------------------+------------------------+
| "newfunc"                     | "PyObject" * "PyObject" *     | "PyObject" *           |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+
| "initproc"                    | "PyObject" * "PyObject" *     | int                    |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+
| "reprfunc"                    | "PyObject" *                  | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "getattrfunc"                 | "PyObject" * const char *     | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "setattrfunc"                 | "PyObject" * const char *     | int                    |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+
| "getattrofunc"                | "PyObject" * "PyObject" *     | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "setattrofunc"                | "PyObject" * "PyObject" *     | int                    |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+
| "descrgetfunc"                | "PyObject" * "PyObject" *     | "PyObject" *           |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+
| "descrsetfunc"                | "PyObject" * "PyObject" *     | int                    |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+
| "hashfunc"                    | "PyObject" *                  | Py_hash_t              |
+-------------------------------+-------------------------------+------------------------+
| "richcmpfunc"                 | "PyObject" * "PyObject" * int | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "getiterfunc"                 | "PyObject" *                  | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "iternextfunc"                | "PyObject" *                  | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "lenfunc"                     | "PyObject" *                  | "Py_ssize_t"           |
+-------------------------------+-------------------------------+------------------------+
| "getbufferproc"               | "PyObject" * "Py_buffer" *    | int                    |
|                               | int                           |                        |
+-------------------------------+-------------------------------+------------------------+
| "releasebufferproc"           | "PyObject" * "Py_buffer" *    | void                   |
+-------------------------------+-------------------------------+------------------------+
| "inquiry"                     | "PyObject" *                  | int                    |
+-------------------------------+-------------------------------+------------------------+
| "unaryfunc"                   | "PyObject" *                  | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "binaryfunc"                  | "PyObject" * "PyObject" *     | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "ternaryfunc"                 | "PyObject" * "PyObject" *     | "PyObject" *           |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+
| "ssizeargfunc"                | "PyObject" * "Py_ssize_t"     | "PyObject" *           |
+-------------------------------+-------------------------------+------------------------+
| "ssizeobjargproc"             | "PyObject" * "Py_ssize_t"     | int                    |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+
| "objobjproc"                  | "PyObject" * "PyObject" *     | int                    |
+-------------------------------+-------------------------------+------------------------+
| "objobjargproc"               | "PyObject" * "PyObject" *     | int                    |
|                               | "PyObject" *                  |                        |
+-------------------------------+-------------------------------+------------------------+

请参阅 槽位类型 typedef 里有更多详细信息。


PyTypeObject 定义
=================

"PyTypeObject" 的结构定义可以在 "Include/object.h" 中找到。 为了方便参
考，此处复述了其中的定义:

   typedef struct _typeobject {
       PyObject_VAR_HEAD
       const char *tp_name; /* For printing, in format "<module>.<name>" */
       Py_ssize_t tp_basicsize, tp_itemsize; /* For allocation */

       /* Methods to implement standard operations */

       destructor tp_dealloc;
       Py_ssize_t tp_vectorcall_offset;
       getattrfunc tp_getattr;
       setattrfunc tp_setattr;
       PyAsyncMethods *tp_as_async; /* formerly known as tp_compare (Python 2)
                                       or tp_reserved (Python 3) */
       reprfunc tp_repr;

       /* Method suites for standard classes */

       PyNumberMethods *tp_as_number;
       PySequenceMethods *tp_as_sequence;
       PyMappingMethods *tp_as_mapping;

       /* More standard operations (here for binary compatibility) */

       hashfunc tp_hash;
       ternaryfunc tp_call;
       reprfunc tp_str;
       getattrofunc tp_getattro;
       setattrofunc tp_setattro;

       /* Functions to access object as input/output buffer */
       PyBufferProcs *tp_as_buffer;

       /* Flags to define presence of optional/expanded features */
       unsigned long tp_flags;

       const char *tp_doc; /* Documentation string */

       /* Assigned meaning in release 2.0 */
       /* call function for all accessible objects */
       traverseproc tp_traverse;

       /* delete references to contained objects */
       inquiry tp_clear;

       /* Assigned meaning in release 2.1 */
       /* rich comparisons */
       richcmpfunc tp_richcompare;

       /* weak reference enabler */
       Py_ssize_t tp_weaklistoffset;

       /* Iterators */
       getiterfunc tp_iter;
       iternextfunc tp_iternext;

       /* Attribute descriptor and subclassing stuff */
       struct PyMethodDef *tp_methods;
       struct PyMemberDef *tp_members;
       struct PyGetSetDef *tp_getset;
       // Strong reference on a heap type, borrowed reference on a static type
       struct _typeobject *tp_base;
       PyObject *tp_dict;
       descrgetfunc tp_descr_get;
       descrsetfunc tp_descr_set;
       Py_ssize_t tp_dictoffset;
       initproc tp_init;
       allocfunc tp_alloc;
       newfunc tp_new;
       freefunc tp_free; /* Low-level free-memory routine */
       inquiry tp_is_gc; /* For PyObject_IS_GC */
       PyObject *tp_bases;
       PyObject *tp_mro; /* method resolution order */
       PyObject *tp_cache;
       PyObject *tp_subclasses;
       PyObject *tp_weaklist;
       destructor tp_del;

       /* Type attribute cache version tag. Added in version 2.6 */
       unsigned int tp_version_tag;

       destructor tp_finalize;
       vectorcallfunc tp_vectorcall;

       /* bitset of which type-watchers care about this type */
       unsigned char tp_watched;
   } PyTypeObject;


PyObject 槽位
=============

类型对象结构体扩展了 "PyVarObject" 结构体。 "ob_size" 字段用于动态类型
（由 "type_new()" 创建，通常由 class 语句调用）。 请注意 "PyType_Type"
（元类型）会初始化 "tp_itemsize"，这意味着它的实例（即类型对象） *必须
* 具有 "ob_size" 字段。

Py_ssize_t PyObject.ob_refcnt
    * 属于 稳定 ABI.*

   这是类型对象的引用计数，由 "PyObject_HEAD_INIT" 宏初始化为 "1"。 请
   注意对于 静态分配的类型对象，类型的实例（其 "ob_type" 指向该类型的
   对象） *不会被* 计入引用。 但对于 动态分配的类型对象，实例 *会被*
   计入引用。

   **继承：**

   子类型不继承此字段。

PyTypeObject *PyObject.ob_type
    * 属于 稳定 ABI.*

   这是类型的类型，换句话说就是元类型，它由宏 "PyObject_HEAD_INIT" 的
   参数来做初始化，它的值一般情况下是 "&PyType_Type" 。可是为了使动态
   可载入扩展模块至少在Windows上可用，编译器会报错这是一个不可用的初始
   化。因此按照惯例传递 "NULL" 给宏 "PyObject_HEAD_INIT" 并且在模块的
   初始化函数开始时候其他任何操作之前初始化这个字段。典型做法是这样的
   ：

      Foo_Type.ob_type = &PyType_Type;

   这应当在创建类型的任何实例之前完成。 "PyType_Ready()" 会检查
   "ob_type" 是否为 "NULL"，如果是，则将其初始化为基类的 "ob_type" 字
   段。 如果该字段为非零值则 "PyType_Ready()" 将不会更改它。

   **继承：**

   此字段会被子类型继承。

PyObject *PyObject._ob_next
PyObject *PyObject._ob_prev

   These fields are only present when the macro "Py_TRACE_REFS" is
   defined (see the "configure --with-trace-refs option").

   Their initialization to "NULL" is taken care of by the
   "PyObject_HEAD_INIT" macro.  For statically allocated objects,
   these fields always remain "NULL".  For dynamically allocated
   objects, these two fields are used to link the object into a doubly
   linked list of *all* live objects on the heap.

   This could be used for various debugging purposes; currently the
   only uses are the "sys.getobjects()" function and to print the
   objects that are still alive at the end of a run when the
   environment variable "PYTHONDUMPREFS" is set.

   **继承：**

   These fields are not inherited by subtypes.


PyVarObject 槽位
================

Py_ssize_t PyVarObject.ob_size
    * 属于 稳定 ABI.*

   对于 静态分配的内存对象，它应该初始化为 0。对于 动态分配的类型对象
   ，该字段具有特殊的内部含义。

   **继承：**

   子类型不继承此字段。


PyTypeObject 槽
===============

每个槽位都有一个小节来描述继承关系。 如果 "PyType_Ready()" 可以在字段
被设为 "NULL" 时设置一个值那么还会有一个“默认”小节。 （请注意在
"PyBaseObject_Type" 和 "PyType_Type" 上设置的许多字段实际上就是默认值
。）

const char *PyTypeObject.tp_name

   指向包含类型名称的以 NUL 结尾的字符串的指针。 对于可作为模块全局访
   问的类型，该字符串应为模块全名，后面跟一个点号，然后再加类型名称；
   对于内置类型，它应当只是类型名称。 如果模块是包的子模块，则包的全名
   将是模块的全名的一部分。 例如，在包 "P" 的子包 "Q" 中的模块 "M" 中
   定义的名为 "T" 的类型应当具有 "tp_name" 初始化器 ""P.Q.M.T""。

   对于 动态分配的类型对象，这应为类型名称，而模块名称将作为
   "'__module__'" 键的值显式地保存在类型字典中。

   对于 静态分配的类型对象，*tp_name* 字段应当包含一个点号。 最后一个
   点号之前的所有内容都可作为 "__module__" 属性访问，而最后一个点号之
   后的所有内容都可作为 "__name__" 属性访问。

   如果不存在点号，则整个 "tp_name" 字段将作为 "__name__" 属性访问，而
   "__module__" 属性则将是未定义的（除非在字典中显式地设置，如上文所述
   ）。 这意味着你的类型将无法执行 pickle。 此外，用 pydoc 创建的模块
   文档中也不会列出该类型。

   该字段不可为 "NULL"。 它是 "PyTypeObject()" 中唯一的必填字段（除了
   潜在的 "tp_itemsize" 以外）。

   **继承：**

   子类型不继承此字段。

Py_ssize_t PyTypeObject.tp_basicsize
Py_ssize_t PyTypeObject.tp_itemsize

   通过这些字段可以计算出该类型实例以字节为单位的大小。

   存在两种类型：具有固定长度实例的类型其 "tp_itemsize" 字段为零；具有
   可变长度实例的类型其 "tp_itemsize" 字段不为零。 对于具有固定长度实
   例的类型，所有实例的大小都相同，具体大小由 "tp_basicsize" 给出。

   对于具有可变长度实例的类型，实例必须有一个 "ob_size" 字段，实例大小
   为 "tp_basicsize" 加上 N 乘以 "tp_itemsize"，其中 N 是对象的“长度”
   。 N 的值通常存储在实例的 "ob_size" 字段中。 但也有例外：举例来说，
   整数类型使用负的 "ob_size" 来表示负数，N 在这里就是 "abs(ob_size)"
   。 此外，在实例布局中存在 "ob_size" 字段并不意味着实例结构是可变长
   度的（例如，列表类型的结构体有固定长度的实例，但这些实例却包含一个
   有意义的 "ob_size" 字段）。

   基本大小包括由宏 "PyObject_HEAD" 或 "PyObject_VAR_HEAD" （以用于声
   明实例结构的宏为准）声明的实例中的字段，如果存在 "_ob_prev" 和
   "_ob_next" 字段则将相应地包括这些字段。 这意味着为 "tp_basicsize"
   获取初始化器的唯一正确方式是在用于声明实例布局的结构上使用 "sizeof"
   操作符。 基本大小不包括 GC 标头的大小。

   关于对齐的说明：如果变量条目需要特定的对齐，则应通过 "tp_basicsize"
   的值来处理。 例如：假设某个类型实现了一个 "double" 数组。
   "tp_itemsize" 就是 "sizeof(double)"。 程序员有责任确保
   "tp_basicsize" 是 "sizeof(double)" 的倍数（假设这是 "double" 的对齐
   要求）。

   对于任何具有可变长度实例的类型，该字段不可为 "NULL"。

   **继承：**

   这些字段将由子类分别继承。 如果基本类型有一个非零的 "tp_itemsize"，
   那么在子类型中将 "tp_itemsize" 设置为不同的非零值通常是不安全的（不
   过这取决于该基本类型的具体实现）。

destructor PyTypeObject.tp_dealloc

   指向实例析构函数的指针。除非保证类型的实例永远不会被释放（就像单例
   对象 "None" 和 "Ellipsis" 那样），否则必须定义这个函数。函数声明如
   下：

      void tp_dealloc(PyObject *self);

   当新引用计数为零时，"Py_DECREF()" 和 "Py_XDECREF()" 宏会调用析构函
   数。 此时，实例仍然存在，但已没有了对它的引用。 析构函数应当释放该
   实例拥有的所有引用，释放实例拥有的所有内存缓冲区（使用与分配缓冲区
   时所用分配函数相对应的释放函数），并调用类型的 "tp_free" 函数。 如
   果该类型不可子类型化（未设置 "Py_TPFLAGS_BASETYPE" 旗标位），则允许
   直接调用对象的释放器而不必通过 "tp_free"。 对象的释放器应为分配实例
   时所使用的释放器；如果实例是使用 "PyObject_New" 或
   "PyObject_NewVar" 分配的，则释放器通常为 "PyObject_Del()"；如果实例
   是使用 "PyObject_GC_New" 或 "PyObject_GC_NewVar" 分配的，则释放器通
   常为 "PyObject_GC_Del()"。

   如果该类型支持垃圾回收（设置了 "Py_TPFLAGS_HAVE_GC" 旗标位），则析
   构器应在清除任何成员字段之前调用 "PyObject_GC_UnTrack()"。

      static void foo_dealloc(foo_object *self) {
          PyObject_GC_UnTrack(self);
          Py_CLEAR(self->ref);
          Py_TYPE(self)->tp_free((PyObject *)self);
      }

   最后，如果该类型是堆分配的 ("Py_TPFLAGS_HEAPTYPE")，则在调用类型释
   放器后，释放器应释放对其类型对象的所有引用 (通过 "Py_DECREF()")。
   为了避免悬空指针，建议的实现方式如下：

      static void foo_dealloc(foo_object *self) {
          PyTypeObject *tp = Py_TYPE(self);
          // free references and buffers here
          tp->tp_free(self);
          Py_DECREF(tp);
      }

   **继承：**

   此字段会被子类型继承。

Py_ssize_t PyTypeObject.tp_vectorcall_offset

   一个相对使用 vectorcall 协议 实现调用对象的实例级函数的可选的偏移量
   ，这是一种比简单的 "tp_call" 更有效的替代选择。

   该字段仅在设置了 "Py_TPFLAGS_HAVE_VECTORCALL" 旗标时使用。 在此情况
   下，它必须为一个包含 "vectorcallfunc" 指针实例中的偏移量的正整数。

   *vectorcallfunc* 指针可能为 "NULL"，在这种情况下实例的行为就像
   "Py_TPFLAGS_HAVE_VECTORCALL" 没有被设置一样：调用实例操作会回退至
   "tp_call"。

   任何设置了 "Py_TPFLAGS_HAVE_VECTORCALL" 的类也必须设置 "tp_call" 并
   确保其行为与 *vectorcallfunc* 函数一致。 这可以通过将 *tp_call* 设
   为 "PyVectorcall_Call()" 来实现。

   在 3.8 版本发生变更: 在 3.8 版之前，这个槽位被命名为 "tp_print"。
   在 Python 2.x 中，它被用于打印到文件。 在 Python 3.0 至 3.7 中，它
   没有被使用。

   在 3.12 版本发生变更: 在 3.12 版之前，不推荐 可变堆类型 实现
   vectorcall 协议。 当用户在 Python 代码中设置 "__call__" 时，只有
   *tp_call* 会被更新，很可能使它与 vectorcall 函数不一致。 自 3.12 起
   ，设置 "__call__" 将通过清除 "Py_TPFLAGS_HAVE_VECTORCALL" 旗标来禁
   用 vectorcall 优化。

   **继承：**

   该字段总是会被继承。 但是，"Py_TPFLAGS_HAVE_VECTORCALL" 旗标并不总
   是会被继承。 如果它未被设置，则子类不会使用 vectorcall，除非显式地
   调用了 "PyVectorcall_Call()"。

getattrfunc PyTypeObject.tp_getattr

   一个指向获取属性字符串函数的可选指针。

   该字段已弃用。当它被定义时，应该和 "tp_getattro" 指向同一个函数，但
   接受一个C字符串参数表示属性名，而不是Python字符串对象。

   **继承：**

   分组: "tp_getattr", "tp_getattro"

   该字段会被子类和 "tp_getattro" 所继承：当子类型的 "tp_getattr" 和
   "tp_getattro" 均为 "NULL" 时该子类型将从它的基类型同时继承
   "tp_getattr" 和 "tp_getattro"。

setattrfunc PyTypeObject.tp_setattr

   一个指向函数以便设置和删除属性的可选指针。

   该字段已弃用。当它被定义时，应该和 "tp_setattro" 指向同一个函数，但
   接受一个C字符串参数表示属性名，而不是Python字符串对象。

   **继承：**

   分组: "tp_setattr", "tp_setattro"

   该字段会被子类型和 "tp_setattro" 所继承：当子类型的 "tp_setattr" 和
   "tp_setattro" 均为 "NULL" 时该子类型将同时从它的基类型继承
   "tp_setattr" 和 "tp_setattro"。

PyAsyncMethods *PyTypeObject.tp_as_async

   指向一个包含仅与在 C 层级上实现 *awaitable* 和 *asynchronous
   iterator* 协议的对象相关联的字段的附加结构体。 请参阅 异步对象结构
   体 了解详情。

   Added in version 3.5: 在之前被称为 "tp_compare" 和 "tp_reserved"。

   **继承：**

   "tp_as_async" 字段不会被继承，但所包含的字段会被单独继承。

reprfunc PyTypeObject.tp_repr

   一个实现了内置函数 "repr()" 的函数的可选指针。

   该签名与 "PyObject_Repr()" 的相同:

      PyObject *tp_repr(PyObject *self);

   该函数必须返回一个字符串或 Unicode 对象。 在理想情况下，该函数应当
   返回一个字符串，当将其传给 "eval()" 时，只要有合适的环境，就会返回
   一个具有相同值的对象。 如果这不可行，则它应当返回一个以 "'<'" 开头
   并以 "'>'" 结尾的可被用来推断出对象的类型和值的字符串。

   **继承：**

   此字段会被子类型继承。

   **默认：**

   如果未设置该字段，则返回 "<%s object at %p>" 形式的字符串，其中
   "%s" 将替换为类型名称，"%p" 将替换为对象的内存地址。

PyNumberMethods *PyTypeObject.tp_as_number

   指向一个附加结构体的指针，其中包含只与执行数字协议的对象相关的字段
   。 这些字段的文档参见 数字对象结构体。

   **继承：**

   "tp_as_number" 字段不会被继承，但所包含的字段会被单独继承。

PySequenceMethods *PyTypeObject.tp_as_sequence

   指向一个附加结构体的指针，其中包含只与执行序列协议的对象相关的字段
   。 这些字段的文档见 序列对象结构体。

   **继承：**

   "tp_as_sequence" 字段不会被继承，但所包含的字段会被单独继承。

PyMappingMethods *PyTypeObject.tp_as_mapping

   指向一个附加结构体的指针，其中包含只与执行映射协议的对象相关的字段
   。 这些字段的文档见 映射对象结构体。

   **继承：**

   "tp_as_mapping" 字段不会继承，但所包含的字段会被单独继承。

hashfunc PyTypeObject.tp_hash

   一个指向实现了内置函数 "hash()" 的函数的可选指针。

   其签名与 "PyObject_Hash()" 的相同:

      Py_hash_t tp_hash(PyObject *);

   "-1" 不应作为正常返回值被返回；当计算哈希值过程中发生错误时，函数应
   设置一个异常并返回 "-1"。

   当该字段（*和* "tp_richcompare"）都未设置，尝试对该对象取哈希会引发
   "TypeError"。这与将其设为 "PyObject_HashNotImplemented()" 相同。

   此字段可被显式设为 "PyObject_HashNotImplemented()" 以阻止从父类型继
   承哈希方法。在 Python 层面这被解释为 "__hash__ = None" 的等价物，使
   得 "isinstance(o, collections.Hashable)" 正确返回 "False".。请注意
   反过来也是如此：在 Python 层面设置一个类的 "__hash__ = None" 会使得
   "tp_hash" 槽位被设置为 "PyObject_HashNotImplemented()"。

   **继承：**

   分组: "tp_hash", "tp_richcompare"

   该字段会被子类型同 "tp_richcompare" 一起继承：当子类型的
   "tp_richcompare" 和 "tp_hash" 均为 "NULL" 时子类型将同时继承
   "tp_richcompare" 和 "tp_hash"。

ternaryfunc PyTypeObject.tp_call

   一个可选的实现对象调用的指向函数的指针。 如果对象不是可调用对象则该
   值应为 "NULL"。 其签名与 "PyObject_Call()" 的相同:

      PyObject *tp_call(PyObject *self, PyObject *args, PyObject *kwargs);

   **继承：**

   此字段会被子类型继承。

reprfunc PyTypeObject.tp_str

   一个可选的实现内置 "str()" 操作的函数的指针。 （请注意 "str" 现在是
   一个类型，"str()" 是调用该类型的构造器。 该构造器将调用
   "PyObject_Str()" 执行实际操作，而 "PyObject_Str()" 将调用该处理器。
   ）

   其签名与 "PyObject_Str()" 的相同:

      PyObject *tp_str(PyObject *self);

   该函数必须返回一个字符串或 Unicode 对象。 它应当是一个“友好”的对象
   字符串表示形式，因为这就是要在 "print()" 函数中与其他内容一起使用的
   表示形式。

   **继承：**

   此字段会被子类型继承。

   **默认：**

   当未设置该字段时，将调用 "PyObject_Repr()" 来返回一个字符串表示形式
   。

getattrofunc PyTypeObject.tp_getattro

   一个指向获取属性字符串函数的可选指针。

   其签名与 "PyObject_GetAttr()" 的相同:

      PyObject *tp_getattro(PyObject *self, PyObject *attr);

   可以方便地将该字段设为 "PyObject_GenericGetAttr()"，它实现了查找对
   象属性的通常方式。

   **继承：**

   分组: "tp_getattr", "tp_getattro"

   该字段会被子类同 "tp_getattr" 一起继承：当子类型的 "tp_getattr" 和
   "tp_getattro" 均为 "NULL" 时子类型将同时继承 "tp_getattr" 和
   "tp_getattro"。

   **默认：**

   "PyBaseObject_Type" 使用 "PyObject_GenericGetAttr()"。

setattrofunc PyTypeObject.tp_setattro

   一个指向函数以便设置和删除属性的可选指针。

   其签名与 "PyObject_SetAttr()" 的相同:

      int tp_setattro(PyObject *self, PyObject *attr, PyObject *value);

   此外，还必须支持将 *value* 设为 "NULL" 来删除属性。 通常可以方便地
   将该字段设为 "PyObject_GenericSetAttr()"，它实现了设备对象属性的通
   常方式。

   **继承：**

   分组: "tp_setattr", "tp_setattro"

   该字段会被子类型同 "tp_setattr" 一起继承：当子类型的 "tp_setattr"
   和 "tp_setattro" 均为 "NULL" 时子类型将同时继承 "tp_setattr" 和
   "tp_setattro"。

   **默认：**

   "PyBaseObject_Type" 使用 "PyObject_GenericSetAttr()"。

PyBufferProcs *PyTypeObject.tp_as_buffer

   指向一个包含只与实现缓冲区接口的对象相关的字段的附加结构体的指针。
   这些字段的文档参见 缓冲区对象结构体。

   **继承：**

   "tp_as_buffer" 字段不会被继承，但所包含的字段会被单独继承。

unsigned long PyTypeObject.tp_flags

   该字段是针对多个旗标的位掩码。 某些旗标指明用于特定场景的变化语义；
   另一些旗标则用于指明类型对象（或通过 "tp_as_number",
   "tp_as_sequence", "tp_as_mapping" 和 "tp_as_buffer" 引用的扩展结构
   体）中的特定字段，它们在历史上并不总是有效；如果这样的旗标位是清晰
   的，则它所保护的类型字段必须不可被访问并且必须被视为具有零或 "NULL"
   值。

   **继承：**

   这个字段的继承很复杂。 大多数旗标位都是单独继承的，也就是说，如果基
   类型设置了一个旗标位，则子类型将继承该旗标位。 从属于扩展结构体的旗
   标位仅在扩展结构体被继承时才会被继承，也就是说，基类型的旗标位值会
   与指向扩展结构体的指针一起被拷贝到子类型中。 "Py_TPFLAGS_HAVE_GC"
   旗标位会与 "tp_traverse" 和 "tp_clear" 字段一起被继承，也就是说，如
   果 "Py_TPFLAGS_HAVE_GC" 旗标位在子类型中被清空并且子类型中的
   "tp_traverse" 和 "tp_clear" 字段存在并具有 "NULL" 值。 .. XXX 那么
   大多数旗标位 *真的* 都是单独继承的吗？

   **默认：**

   "PyBaseObject_Type" 使用 "Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE"
   。

   **位掩码:**

   目前定义了以下位掩码；可以使用 "|" 运算符对它们进行 OR 运算以形成
   "tp_flags" 字段的值。 宏 "PyType_HasFeature()" 接受一个类型和一个旗
   标值 *tp* 和 *f*，并检查 "tp->tp_flags & f" 是否为非零值。

   Py_TPFLAGS_HEAPTYPE

      当类型对象本身在堆上被分配时会设置这个比特位，例如，使用
      "PyType_FromSpec()" 动态创建的类型。 在此情况下，其实例的
      "ob_type" 字段会被视为指向该类型的引用，而类型对象将在一个新实例
      被创建时执行 INCREF，并在实例被销毁时执行 DECREF（这不会应用于子
      类型的实例；只有实例的 ob_type 所引用的类型会执行 INCREF 和
      DECREF）。 堆类型应当也 支持垃圾回收 因为它们会形成对它们自己的
      模块对象的循环引用。

      **继承：**

      ？？？

   Py_TPFLAGS_BASETYPE

      当此类型可被用作另一个类型的基类型时该比特位将被设置。 如果该比
      特位被清除，则此类型将无法被子类型化（类似于 Java 中的 "final"
      类）。

      **继承：**

      ？？？

   Py_TPFLAGS_READY

      当此类型对象通过 "PyType_Ready()" 被完全实例化时该比特位将被设置
      。

      **继承：**

      ？？？

   Py_TPFLAGS_READYING

      当 "PyType_Ready()" 处在初始化此类型对象过程中时该比特位将被设置
      。

      **继承：**

      ？？？

   Py_TPFLAGS_HAVE_GC

      当对象支持垃圾回收时会设置这个旗标位。 如果设置了这个位，则实例
      必须使用 "PyObject_GC_New" 来创建并使用 "PyObject_GC_Del()" 来销
      毁。 更多信息参见 使对象类型支持循环垃圾回收。 这个位还会假定类
      型对象中存在 GC 相关字段 "tp_traverse" 和 "tp_clear"。

      **继承：**

      分组: "Py_TPFLAGS_HAVE_GC", "tp_traverse", "tp_clear"

      "Py_TPFLAGS_HAVE_GC" 旗标位会与 "tp_traverse" 和 "tp_clear" 字段
      一起被继承，也就是说，如果 "Py_TPFLAGS_HAVE_GC" 旗标位在子类型中
      被清空并且子类型中的 "tp_traverse" 和 "tp_clear" 字段存在并具有
      "NULL" 值的话。

   Py_TPFLAGS_DEFAULT

      这是一个从属于类型对象及其扩展结构体的存在的所有位的位掩码。 目
      前，它包括以下的位: "Py_TPFLAGS_HAVE_STACKLESS_EXTENSION"。

      **继承：**

      ？？？

   Py_TPFLAGS_METHOD_DESCRIPTOR

      这个位指明对象的行为类似于未绑定方法。

      如果为 "type(meth)" 设置了该旗标，那么：

      * "meth.__get__(obj, cls)(*args, **kwds)" (其中 "obj" 不为 None)
        必须等价于 "meth(obj, *args, **kwds)"。

      * "meth.__get__(None, cls)(*args, **kwds)" 必须等价于
        "meth(*args, **kwds)"。

      此旗标为 "obj.meth()" 这样的典型方法调用启用优化：它将避免为
      "obj.meth" 创建临时的“绑定方法”对象。

      Added in version 3.8.

      **继承：**

      此旗标绝不会被没有设置 "Py_TPFLAGS_IMMUTABLETYPE" 旗标的类型所继
      承。 对于扩展类型，当 "tp_descr_get" 被继承时它也会被继承。

   Py_TPFLAGS_MANAGED_DICT

      该比特位表示类的实例具有 "__dict__" 属性，并且该字典的空间是由
      VM 管理的。

      如果设置了该旗标，则 "Py_TPFLAGS_HAVE_GC" 也应当被设置。

      Added in version 3.12.

      **继承：**

      此旗标将被继承，除非某个超类设置了 "tp_dictoffset" 字段。

   Py_TPFLAGS_MANAGED_WEAKREF

      该比特位表示类的实例应当是可被弱引用的。

      Added in version 3.12.

      **继承：**

      此旗标将被继承，除非某个超类设置了 "tp_weaklistoffset" 字段。

   Py_TPFLAGS_ITEMS_AT_END

      仅适用于可变大小的类型，也就是说，具有非零 "tp_itemsize" 值的类
      型。

      表示此类型的实例的可变大小部分位于该实例内存区的末尾，其偏移量为
      "Py_TYPE(obj)->tp_basicsize" (每个子类可能不一样)。

      当设置此旗标时，请确保所有子类要么使用此内存布局，要么不是可变大
      小。 Python 不会检查这一点。

      Added in version 3.12.

      **继承：**

      这个旗标会被继承。

   Py_TPFLAGS_LONG_SUBCLASS

   Py_TPFLAGS_LIST_SUBCLASS

   Py_TPFLAGS_TUPLE_SUBCLASS

   Py_TPFLAGS_BYTES_SUBCLASS

   Py_TPFLAGS_UNICODE_SUBCLASS

   Py_TPFLAGS_DICT_SUBCLASS

   Py_TPFLAGS_BASE_EXC_SUBCLASS

   Py_TPFLAGS_TYPE_SUBCLASS

      这些旗标被 "PyLong_Check()" 等函数用来快速确定一个类型是否为内置
      类型的子类；这样的专用检测比泛用检测如 "PyObject_IsInstance()"
      要更快速。 继承自内置类型的自定义类型应当正确地设置其 "tp_flags"
      ，否则与这样的类型进行交互的代码将因所使用的检测种类而出现不同的
      行为。

   Py_TPFLAGS_HAVE_FINALIZE

      当类型结构体中存在 "tp_finalize" 槽位时会设置这个比特位。

      Added in version 3.4.

      自 3.8 版本弃用: 此旗标已不再是必要的，因为解释器会假定类型结构
      体中总是存在 "tp_finalize" 槽位。

   Py_TPFLAGS_HAVE_VECTORCALL

      当类实现了 vectorcall 协议 时会设置这个比特位。 请参阅
      "tp_vectorcall_offset" 了解详情。

      **继承：**

      如果继承了 "tp_call" 则也会继承这个比特位。

      Added in version 3.9.

      在 3.12 版本发生变更: 现在当类的 "__call__()" 方法被重新赋值时该
      旗标将从类中移除。现在该旗标能被可变类所继承。

   Py_TPFLAGS_IMMUTABLETYPE

      不可变的类型对象会设置这个比特位：类型属性无法被设置或删除。

      "PyType_Ready()" 会自动对 静态类型 应用这个旗标。

      **继承：**

      这个旗标不会被继承。

      Added in version 3.10.

   Py_TPFLAGS_DISALLOW_INSTANTIATION

      不允许创建此类型的实例：将 "tp_new" 设为 NULL 并且不会在类型字符
      中创建 "__new__" 键。

      这个旗标必须在创建该类型之前设置，而不是在之后。 例如，它必须在
      该类型调用 "PyType_Ready()" 之前被设置。

      如果 "tp_base" 为 NULL 或者 "&PyBaseObject_Type" 和 "tp_new" 为
      NULL 则该旗标会在 静态类型 上自动设置。

      **继承：**

      这个旗标不会被继承。 但是，子类将不能被实例化，除非它们提供了不
      为 NULL 的 "tp_new" (这只能通过 C API 实现)。

      备注:

        要禁止直接实例化一个类但允许实例化其子类 (例如对于 *abstract
        base class*)，请勿使用此旗标。 替代的做法是，让 "tp_new" 只对
        子类可用。

      Added in version 3.10.

   Py_TPFLAGS_MAPPING

      这个比特位指明该类的实例可以在被用作 "match" 代码块的目标时匹配
      映射模式。 它会在注册或子类化 "collections.abc.Mapping" 时自动设
      置，并在注册 "collections.abc.Sequence" 时取消设置。

      备注:

        "Py_TPFLAGS_MAPPING" 和 "Py_TPFLAGS_SEQUENCE" 是互斥的；同时启
        用两个旗标将导致报错。

      **继承：**

      这个旗标将被尚未设置 "Py_TPFLAGS_SEQUENCE" 的类型所继承。

      参见: **PEP 634** —— 结构化模式匹配：规范

      Added in version 3.10.

   Py_TPFLAGS_SEQUENCE

      这个比特位指明该类的实例可以在被用作 "match" 代码块的目标时匹配
      序列模式。 它会在注册或子类化 "collections.abc.Sequence" 时自动
      设置，并在注册 "collections.abc.Mapping" 时取消设置。

      备注:

        "Py_TPFLAGS_MAPPING" 和 "Py_TPFLAGS_SEQUENCE" 是互斥的；同时启
        用两个旗标将导致报错。

      **继承：**

      这个旗标将被尚未设置 "Py_TPFLAGS_MAPPING" 的类型所继承。

      参见: **PEP 634** —— 结构化模式匹配：规范

      Added in version 3.10.

   Py_TPFLAGS_VALID_VERSION_TAG

      内部使用。 请不要设置或取消设置此旗标。 用于指明一个类具有被修改
      的调用 "PyType_Modified()"

      警告:

        这个旗标存在于头文件中，但是属于内部特性而不应直接使用。 它将
        在未来的 CPython 版本中被移除

const char *PyTypeObject.tp_doc

   一个可选的指向给出该类型对象的文档字符串的以 NUL 结束的 C 字符串的
   指针。 该指针被暴露为类型和类型实例上的 "__doc__" 属性。

   **继承：**

   这个字段 *不会* 被子类型继承。

traverseproc PyTypeObject.tp_traverse

   一个可选的指向针对垃圾回收器的遍历函数的指针。 该指针仅会在设置了
   "Py_TPFLAGS_HAVE_GC" 旗标位时被使用。 函数签名为:

      int tp_traverse(PyObject *self, visitproc visit, void *arg);

   有关 Python 垃圾回收方案的更多信息可在 使对象类型支持循环垃圾回收
   一节中查看。

   "tp_traverse" 指针被垃圾回收器用来检测循环引用。 "tp_traverse" 函数
   的典型实现会在实例的每个属于该实例所拥有的 Python 对象的成员上简单
   地调用 "Py_VISIT()"。 例如，以下是来自 "_thread" 扩展模块的函数
   "local_traverse()":

      static int
      local_traverse(localobject *self, visitproc visit, void *arg)
      {
          Py_VISIT(self->args);
          Py_VISIT(self->kw);
          Py_VISIT(self->dict);
          return 0;
      }

   请注意 "Py_VISIT()" 仅能在可以参加循环引用的成员上被调用。 虽然还存
   在一个 "self->key" 成员，但它只能为 "NULL" 或 Python 字符串因而不能
   成为循环引用的一部分。

   在另一方面，即使你知道某个成员永远不会成为循环引用的一部分，作为调
   试的辅助你仍然可能想要访问它因此 "gc" 模块的 "get_referents()" 函数
   将会包括它。

   警告:

     当实现 "tp_traverse" 时，只有实例所 *拥有* 的成员 (就是有指向它们
     的 *强引用*) 才必须被访问。 举例来说，如果一个对象通过
     "tp_weaklist" 槽位支持弱引用，那么支持链表 (*tp_weaklist* 所指向
     的对象) 的指针就 **不能** 被访问因为实例并不直接拥有指向自身的弱
     引用 (弱引用列表被用来支持弱引用机制，但实例没有指向其中的元素的
     强引用，因为即使实例还存在它们也允许被删除)。

   Note that "Py_VISIT()" requires the *visit* and *arg* parameters to
   "local_traverse()" to have these specific names; don't name them
   just anything.

   堆分配类型 的实例会持有一个指向其类型的引用。 因此它们的遍历函数必
   须要么访问 "Py_TYPE(self)"，要么通过调用其他堆分配类型（例如一个堆
   分配超类）的 "tp_traverse" 将此任务委托出去。 如果没有这样做，类型
   对象可能不会被垃圾回收。

   在 3.9 版本发生变更: 堆分配类型应当访问 "tp_traverse" 中的
   "Py_TYPE(self)"。 在较早的 Python 版本中，由于 bug 40217，这样做可
   能会导致在超类中发生崩溃。

   **继承：**

   分组: "Py_TPFLAGS_HAVE_GC", "tp_traverse", "tp_clear"

   该字段会与 "tp_clear" 和 "Py_TPFLAGS_HAVE_GC" 旗标位一起被子类型所
   继承：如果旗标位, "tp_traverse" 和 "tp_clear" 在子类型中均为零则它
   们都将从基类型继承。

inquiry PyTypeObject.tp_clear

   一个可选的指向针对垃圾回收器的清理函数的指针。 该指针仅会在设置了
   "Py_TPFLAGS_HAVE_GC" 旗标位时被使用。 函数签名为:

      int tp_clear(PyObject *);

   "tp_clear" 成员函数被用来打破垃圾回收器在循环垃圾中检测到的循环引用
   。 总的来说，系统中的所有 "tp_clear" 函数必须合到一起以打破所有引用
   循环。 这是个微妙的问题，并且如有任何疑问都需要提供 "tp_clear" 函数
   。 例如，元组类型不会实现 "tp_clear" 函数，因为有可能证明完全用元组
   是不会构成循环引用的。 因此其他类型的 "tp_clear" 函数必须足以打破任
   何包含元组的循环。 这不是立即能明确的，并且很少会有避免实现
   "tp_clear" 的适当理由。

   "tp_clear" 的实现应当丢弃实例指向其成员的可能为 Python 对象的引用，
   并将指向这些成员的指针设为 "NULL"，如下面的例子所示:

      static int
      local_clear(localobject *self)
      {
          Py_CLEAR(self->key);
          Py_CLEAR(self->args);
          Py_CLEAR(self->kw);
          Py_CLEAR(self->dict);
          return 0;
      }

   应当使用 "Py_CLEAR()" 宏，因为清除引用是很微妙的：指向被包含对象的
   引用必须在指向被包含对象的指针被设为 "NULL" 之后才能被释放 (通过
   "Py_DECREF()")。 这是因为释放引用可能会导致被包含的对象变成垃圾，触
   发一连串的回收活动，其中可能包括发起调用任意 Python 代码 (由于关联
   到被包含对象的终结器或弱引用回调)。 如果这样的代码有可能再次引用
   *self*，那么这时指向被包含对象的指针为 "NULL" 就是非常重要的，这样
   *self* 就知道被包含对象不可再被使用。 "Py_CLEAR()" 宏将以安全的顺序
   执行此操作。

   请注意 "tp_clear" 并非 *总是* 在实例被取消分配之前被调用。 例如，当
   引用计数足以确定对象不再被使用时，就不会涉及循环垃圾回收器而是直接
   调用 "tp_dealloc"。

   因为 "tp_clear" 函数的目的是打破循环引用，所以不需要清除所包含的对
   象如 Python 字符串或 Python 整数，它们无法参与循环引用。 另一方面，
   清除所包含的全部 Python 对象，并编写类型的 "tp_dealloc" 函数来发起
   调用 "tp_clear" 也很方便。

   有关 Python 垃圾回收方案的更多信息可在 使对象类型支持循环垃圾回收
   一节中查看。

   **继承：**

   分组: "Py_TPFLAGS_HAVE_GC", "tp_traverse", "tp_clear"

   该字段会与 "tp_traverse" 和 "Py_TPFLAGS_HAVE_GC" 旗标位一起被子类型
   所继承：如果旗标位, "tp_traverse" 和 "tp_clear" 在子类型中均为零则
   它们都将从基类型继承。

richcmpfunc PyTypeObject.tp_richcompare

   一个可选的指向富比较函数的指针，函数的签名为:

      PyObject *tp_richcompare(PyObject *self, PyObject *other, int op);

   第一个形参将保证为 "PyTypeObject" 所定义的类型的实例。

   该函数应当返回比较的结果 (通常为 "Py_True" 或 "Py_False")。 如果未
   定义比较运算，它必须返回 "Py_NotImplemented"，如果发生了其他错误则
   它必须返回 "NULL" 并设置一个异常条件。

   以下常量被定义用作 "tp_richcompare" 和 "PyObject_RichCompare()" 的
   第三个参数：

   +----------------------+--------------+
   | 常量                 | 对照         |
   |======================|==============|
   | Py_LT                | "<"          |
   +----------------------+--------------+
   | Py_LE                | "<="         |
   +----------------------+--------------+
   | Py_EQ                | "=="         |
   +----------------------+--------------+
   | Py_NE                | "!="         |
   +----------------------+--------------+
   | Py_GT                | ">"          |
   +----------------------+--------------+
   | Py_GE                | ">="         |
   +----------------------+--------------+

   定义以下宏是为了简化编写丰富的比较函数：

   Py_RETURN_RICHCOMPARE(VAL_A, VAL_B, op)

      从该函数返回 "Py_True" 或 "Py_False"，这取决于比较的结果。 VAL_A
      和 VAL_B 必须是可通过 C 比较运算符进行排序的（例如，它们可以为 C
      整数或浮点数）。 第三个参数指明所请求的运算，与
      "PyObject_RichCompare()" 的参数一样。

      返回值是一个新的 *strong reference*。

      发生错误时，将设置异常并从该函数返回 "NULL"。

      Added in version 3.7.

   **继承：**

   分组: "tp_hash", "tp_richcompare"

   该字段会被子类型同 "tp_hash" 一起继承：当子类型的 "tp_richcompare"
   和 "tp_hash" 均为 "NULL" 时子类型将同时继承 "tp_richcompare" 和
   "tp_hash"。

   **默认：**

   "PyBaseObject_Type" 提供了一个 "tp_richcompare" 的实现，它可以被继
   承。 但是，如果只定义了 "tp_hash"，则不会使用被继承的函数并且该类型
   的实例将无法参加任何比较。

Py_ssize_t PyTypeObject.tp_weaklistoffset

   虽然此字段仍然受到支持，但是如果可能就应当改用
   "Py_TPFLAGS_MANAGED_WEAKREF"。

   如果此类型的实例是可被弱引用的，则该字段将大于零并包含在弱引用列表
   头的实例结构体中的偏移量（忽略 GC 头，如果存在的话）；该偏移量将被
   "PyObject_ClearWeakRefs()" 和 "PyWeakref_*" 函数使用。 实例结构体需
   要包括一个 PyObject* 类型的字段并初始化为 "NULL"。

   不要将该字段与 "tp_weaklist" 混淆；后者是指向类型对象本身的弱引用的
   列表头。

   同时设置 "Py_TPFLAGS_MANAGED_WEAKREF" 位和 "tp_weaklist" 将导致报错
   。

   **继承：**

   该字段会被子类型继承，但注意参阅下面列出的规则。 子类型可以覆盖此偏
   移量；这意味着子类型将使用不同于基类型的弱引用列表。 由于列表头总是
   通过 "tp_weaklistoffset" 找到的，所以这应该不成问题。

   **默认：**

   如果在 "tp_dict" 字段中设置了 "Py_TPFLAGS_MANAGED_WEAKREF" 位，则
   "tp_weaklistoffset" 将被设为负值，用以表明使用此字段是不安全的。

getiterfunc PyTypeObject.tp_iter

   一个可选的指向函数的指针，该函数返回对象的 *iterator*。 它的存在通
   常表明该类型的实例为 *iterable* (尽管序列在没有此函数的情况下也可能
   为可迭代对象)。

   此函数的签名与 "PyObject_GetIter()" 的相同:

      PyObject *tp_iter(PyObject *self);

   **继承：**

   此字段会被子类型继承。

iternextfunc PyTypeObject.tp_iternext

   一个可选的指向函数的指针，该函数返回 *iterator* 中的下一项。 其签名
   为:

      PyObject *tp_iternext(PyObject *self);

   当该迭代器被耗尽时，它必须返回 "NULL"；"StopIteration" 异常可能会设
   置也可能不设置。 当发生另一个错误时，它也必须返回 "NULL"。 它的存在
   表明该类型的实际是迭代器。

   迭代器类型也应当定义 "tp_iter" 函数，并且该函数应当返回迭代器实例本
   身（而不是新的迭代器实例）。

   此函数的签名与 "PyIter_Next()" 的相同。

   **继承：**

   此字段会被子类型继承。

struct PyMethodDef *PyTypeObject.tp_methods

   一个可选的指向 "PyMethodDef" 结构体的以 "NULL" 结束的静态数组的指针
   ，它声明了此类型的常规方法。

   对于该数组中的每一项，都会向类型的字典 (参见下面的 "tp_dict") 添加
   一个包含方法描述器的条目。

   **继承：**

   该字段不会被子类型所继承（方法是通过不同的机制来继承的）。

struct PyMemberDef *PyTypeObject.tp_members

   一个可选的指向 "PyMemberDef" 结构体的以 "NULL" 结束的静态数组的指针
   ，它声明了此类型的常规数据成员（字段或槽位）。

   对于该数组中的每一项，都会向类型的字典 (参见下面的 "tp_dict") 添加
   一个包含方法描述器的条目。

   **继承：**

   该字段不会被子类型所继承（成员是通过不同的机制来继承的）。

struct PyGetSetDef *PyTypeObject.tp_getset

   一个可选的指向 "PyGetSetDef" 结构体的以 "NULL" 结束的静态数组的指针
   ，它声明了此类型的实例中的被计算属性。

   对于该数组中的每一项，都会向类型的字典 (参见下面的 "tp_dict") 添加
   一个包含读写描述器的条目。

   **继承：**

   该字段不会被子类型所继承（被计算属性是通过不同的机制来继承的）。

PyTypeObject *PyTypeObject.tp_base

   一个可选的指向类型特征属性所继承的基类型的指针。 在这个层级上，只支
   持单继承；多重继承需要通过调用元类型动态地创建类型对象。

   备注:

     槽位初始化需要遵循初始化全局变量的规则。 C99 要求初始化器为“地址
     常量”。 隐式转换为指针的函数指示器如 "PyType_GenericNew()" 都是有
     效的 C99 地址常量。但是，生成地址常量并不需要应用于非静态变量如
     "PyBaseObject_Type" 的单目运算符 '&'。 编译器可能支持该运算符（如
     gcc），但 MSVC 则不支持。 这两种编译器在这一特定行为上都是严格符
     合标准的。因此，应当在扩展模块的初始化函数中设置 "tp_base"。

   **继承：**

   该字段不会被子类型继承（显然）。

   **默认：**

   该字段默认为 "&PyBaseObject_Type" (对 Python 程序员来说即 "object"
   类型)。

PyObject *PyTypeObject.tp_dict

   类型的字典将由 "PyType_Ready()" 存储到这里。

   该字段通常应当在 PyType_Ready 被调用之前初始化为 "NULL"；它也可以初
   始化为一个包含类型初始属性的字典。 一旦 "PyType_Ready()" 完成类型的
   初始化，该类型的额外属性只有在它们不与被重载的操作 (如 "__add__()")
   相对应的情况下才会被添加到该字典中。 一旦类型的初始化结束，该字段就
   应被视为是只读的。

   某些类型不会将它们的字典存储在该槽位中。 请使用 "PyType_GetDict()"
   来获取任意类型对应的字典。

   在 3.12 版本发生变更: 内部细节：对于静态内置类型，该值总是为 "NULL"
   。 这种类型的字典是存储在 "PyInterpreterState" 中。 请使用
   "PyType_GetDict()" 来获取任意类型的字典。

   **继承：**

   该字段不会被子类型所继承（但在这里定义的属性是通过不同的机制来继承
   的）。

   **默认：**

   如果该字段为 "NULL"，"PyType_Ready()" 将为它分配一个新字典。

   警告:

     通过字典 C-API 使用 "PyDict_SetItem()" 或修改 "tp_dict" 是不安全
     的。

descrgetfunc PyTypeObject.tp_descr_get

   一个可选的指向“描述器获取”函数的指针。

   函数的签名为:

      PyObject * tp_descr_get(PyObject *self, PyObject *obj, PyObject *type);

   **继承：**

   此字段会被子类型继承。

descrsetfunc PyTypeObject.tp_descr_set

   一个指向用于设置和删除描述器值的函数的选项指针。

   函数的签名为:

      int tp_descr_set(PyObject *self, PyObject *obj, PyObject *value);

   将 *value* 参数设为 "NULL" 以删除该值。

   **继承：**

   此字段会被子类型继承。

Py_ssize_t PyTypeObject.tp_dictoffset

   虽然此字段仍然受到支持，但是如果可能就应当改用
   "Py_TPFLAGS_MANAGED_DICT"。

   如果该类型的实例具有一个包含实例变量的字典，则此字段将为非零值并包
   含该实例变量字典的类型的实例的偏移量；该偏移量将由
   "PyObject_GenericGetAttr()" 使用。

   不要将该字段与 "tp_dict" 混淆；后者是由类型对象本身的属性组成的字典
   。

   该值指定字典相对实例结构体开始位置的偏移量。

   "tp_dictoffset" 应当被视为是只读的。 用于获取指向字典调用
   "PyObject_GenericGetDict()" 的指针。 调用
   "PyObject_GenericGetDict()" 可能需要为字典分配内存，因此在访问对象
   上的属性时调用 "PyObject_GetAttr()" 可能会更有效率。

   同时设置 "Py_TPFLAGS_MANAGED_WEAKREF" 位和 "tp_dictoffset" 将导致报
   错。

   **继承：**

   该字段会被子类型所继承。 子类型不应重写这个偏移量；这样做是不安全的
   ，如果 C 代码试图在之前的偏移量上访问字典的话。 要正确地支持继承，
   请使用 "Py_TPFLAGS_MANAGED_DICT"。

   **默认：**

   这个槽位没有默认值。 对于 静态类型，如果该字段为 "NULL" 则不会为实
   例创建 "__dict__"。

   If the "Py_TPFLAGS_MANAGED_DICT" bit is set in the "tp_dict" field,
   then "tp_dictoffset" will be set to "-1", to indicate that it is
   unsafe to use this field.

initproc PyTypeObject.tp_init

   一个可选的指向实例初始化函数的指针。

   此函数对应于类的 "__init__()" 方法。 和 "__init__()" 一样，创建实例
   时不调用 "__init__()" 是有可能的，并且通过再次调用实例的
   "__init__()" 方法将其重新初始化也是有可能的。

   函数的签名为:

      int tp_init(PyObject *self, PyObject *args, PyObject *kwds);

   self 参数是将要初始化的实例；*args* 和 *kwds* 参数代表调用
   "__init__()" 时传入的位置和关键字参数。

   "tp_init" 函数如果不为 "NULL"，将在通过调用类型正常创建其实例时被调
   用，即在类型的 "tp_new" 函数返回一个该类型的实例时。 如果 "tp_new"
   函数返回了一个不是原始类型的子类型的其他类型的实例，则 "tp_init" 函
   数不会被调用；如果 "tp_new" 返回了一个原始类型的子类型的实例，则该
   子类型的 "tp_init" 将被调用。

   成功时返回 "0"，发生错误时则返回 "-1" 并在错误上设置一个异常。and
   sets an exception on error.

   **继承：**

   此字段会被子类型继承。

   **默认：**

   对于 静态类型 来说该字段没有默认值。

allocfunc PyTypeObject.tp_alloc

   指向一个实例分配函数的可选指针。

   函数的签名为:

      PyObject *tp_alloc(PyTypeObject *self, Py_ssize_t nitems);

   **继承：**

   该字段会被静态子类型继承，但不会被动态子类型（通过 class 语句创建的
   子类型）继承。

   **默认：**

   对于动态子类型，该字段总是会被设为 "PyType_GenericAlloc()"，以强制
   应用标准的堆分配策略。

   对于静态子类型，"PyBaseObject_Type" 将使用 "PyType_GenericAlloc()"
   。 这是适用于所有静态定义类型的推荐值。

newfunc PyTypeObject.tp_new

   一个可选的指向实例创建函数的指针。

   函数的签名为:

      PyObject *tp_new(PyTypeObject *subtype, PyObject *args, PyObject *kwds);

   *subtype* 参数是被创建的对象的类型；*args* 和 *kwds* 参数表示调用类
   型时传入的位置和关键字参数。 请注意 *subtype* 不是必须与被调用的
   "tp_new" 函数所属的类型相同；它可以是该类型的子类型（但不能是完全无
   关的类型）。

   "tp_new" 函数应当调用 "subtype->tp_alloc(subtype, nitems)" 来为对象
   分配空间，然后只执行绝对有必要的进一步初始化操作。 可以安全地忽略或
   重复的初始化操作应当放在 "tp_init" 处理器中。 一个关键的规则是对于
   不可变类型来说，所有初始化操作都应当在 "tp_new" 中发生，而对于可变
   类型，大部分初始化操作都应当推迟到 "tp_init" 再执行。

   设置 "Py_TPFLAGS_DISALLOW_INSTANTIATION" 旗标以禁止在 Python 中创建
   该类型的实例。

   **继承：**

   该字段会被子类型所继承，例外情况是它不会被 "tp_base" 为 "NULL" 或
   "&PyBaseObject_Type" 的 静态类型 所继承。

   **默认：**

   对于 静态类型 该字段没有默认值。 这意味着如果槽位被定义为 "NULL"，
   则无法调用此类型来创建新的实例；应当存在 其他办法来创建实例，例如工
   厂函数等。

freefunc PyTypeObject.tp_free

   一个可选的指向实例释放函数的指针。 函数的签名为:

      void tp_free(void *self);

   一个兼容该签名的初始化器是 "PyObject_Free()"。

   **继承：**

   该字段会被静态子类型继承，但不会被动态子类型（通过 class 语句创建的
   子类型）继承

   **默认：**

   在动态子类型中，该字段会被设为一个适合与 "PyType_GenericAlloc()" 以
   及 "Py_TPFLAGS_HAVE_GC" 旗标位的值相匹配的释放器。

   对于静态子类型，"PyBaseObject_Type" 将使用 "PyObject_Del()"。

inquiry PyTypeObject.tp_is_gc

   可选的指向垃圾回收器所调用的函数的指针。

   垃圾回收器需要知道某个特定的对象是否可以被回收。在一般情况下，垃圾
   回收器只需要检查这个对象类型的 "tp_flags" 字段、以及
   "Py_TPFLAGS_HAVE_GC" 标识位即可做出判断；但是有一些类型同时混合包含
   了静态和动态分配的实例，其中静态分配的实例不应该也无法被回收。本函
   数为后者情况而设计：对于可被垃圾回收的实例，本函数应当返回 "1" ；对
   于不可被垃圾回收的实例，本函数应当返回 "0" 。函数的签名为:

      int tp_is_gc(PyObject *self);

   （此对象的唯一样例是类型本身。 元类型 "PyType_Type" 定义了该函数来
   区分静态和 动态分配的类型。）

   **继承：**

   此字段会被子类型继承。

   **默认：**

   此槽位没有默认值。 如果该字段为 "NULL"，则将使用
   "Py_TPFLAGS_HAVE_GC" 作为相同功能的替代。

PyObject *PyTypeObject.tp_bases

   基类型的元组。

   此字段应当被设为 "NULL" 并被视为只读。 Python 将在类型 "初始化时"
   填充它。

   对于动态创建的类，可以使用 "Py_tp_bases" "槽位" 来代替
   "PyType_FromSpecWithBases()" 的 *bases* 参数。 推荐使用参数形式。

   警告:

     多重继承不适合静态定义的类型。 如果你将 "tp_bases" 设为一个元组，
     Python 将不会引发错误，但某些槽位将只从第一个基类型继承。

   **继承：**

   这个字段不会被继承。

PyObject *PyTypeObject.tp_mro

   包含基类型的扩展集的元组，以类型本身开始并以 "object" 作为结束，使
   用方法解析顺序。

   此字段应当被设为 "NULL" 并被视为只读。 Python 将在类型 "初始化时"
   填充它。

   **继承：**

   这个字段不会被继承；它是通过 "PyType_Ready()" 计算得到的。

PyObject *PyTypeObject.tp_cache

   尚未使用。 仅供内部使用。

   **继承：**

   这个字段不会被继承。

void *PyTypeObject.tp_subclasses

   一组子类。 仅限内部使用的。 可能为无效的指针。

   要获取子类的列表。 请调用 Python 方法 "__subclasses__()"。

   在 3.12 版本发生变更: 对于某些类型，该字段将不带有效的 PyObject*。
   类型已被改为 void* 以指明这一点。

   **继承：**

   这个字段不会被继承。

PyObject *PyTypeObject.tp_weaklist

   弱引用列表头，用于指向该类型对象的弱引用。 不会被继承。 仅限内部使
   用。

   在 3.12 版本发生变更: 内部细节：对于静态内置类型这将总是为 "NULL"，
   即使添加了弱引用也是如此。 每个弱引用都转而保存在
   "PyInterpreterState" 上。 请使用公共 C-API 或内部
   "_PyObject_GET_WEAKREFS_LISTPTR()" 宏来避免此差异。

   **继承：**

   这个字段不会被继承。

destructor PyTypeObject.tp_del

   该字段已被弃用。 请改用 "tp_finalize"。

unsigned int PyTypeObject.tp_version_tag

   用于索引至方法缓存。 仅限内部使用。

   **继承：**

   这个字段不会被继承。

destructor PyTypeObject.tp_finalize

   一个可选的指向实例最终化函数的指针。 函数的签名为:

      void tp_finalize(PyObject *self);

   如果设置了 "tp_finalize"，解释器将在最终化特定实例时调用它一次。 它
   将由垃圾回收器调用（如果实例是单独循环引用的一部分）或是在对象被释
   放之前被调用。 不论是哪种方式，它都肯定会在尝试打破循环引用之前被调
   用，以确保它所操作的对象处于正常状态。

   "tp_finalize" 不应改变当前异常状态；因此，编写非关键终结器的推荐做
   法如下:

      static void
      local_finalize(PyObject *self)
      {
          PyObject *error_type, *error_value, *error_traceback;

          /* Save the current exception, if any. */
          PyErr_Fetch(&error_type, &error_value, &error_traceback);

          /* ... */

          /* Restore the saved exception. */
          PyErr_Restore(error_type, error_value, error_traceback);
      }

   另外还需要注意，在应用垃圾回收机制的 Python 中，"tp_dealloc" 可以从
   任意 Python 线程被调用，而不仅是创建该对象的线程（如果对象成为引用
   计数循环的一部分，则该循环可能会被任何线程上的垃圾回收操作所回收）
   。 这对 Python API 调用来说不是问题，因为 tp_dealloc 调用所在的线程
   将持有全局解释器锁（GIL）。 但是，如果被销毁的对象又销毁了来自其他
   C 或 C++ 库的对象，则应当小心确保在调用 tp_dealloc 的线程上销毁这些
   对象不会破坏这些库的任何资源。

   **继承：**

   此字段会被子类型继承。

   Added in version 3.4.

   在 3.8 版本发生变更: 在 3.8 版之前必须设置
   "Py_TPFLAGS_HAVE_FINALIZE" 旗标才能让该字段被使用。 现在已不再需要
   这样做。

   参见: "安全的对象最终化" (**PEP 442**)

vectorcallfunc PyTypeObject.tp_vectorcall

   用于此类型对象的调用的 vectorcall 函数。 换句话说，它是被用来实现
   "type.__call__" 的 vectorcall。 如果 "tp_vectorcall" 为 "NULL"，默
   认调用实现将使用 "__new__()" 并且 "__init__()" 将被使用。

   **继承：**

   这个字段不会被继承。

   Added in version 3.9: （这个字段从 3.8 起即存在，但是从 3.9 开始投
   入使用）

unsigned char PyTypeObject.tp_watched

   内部对象。 请勿使用。

   Added in version 3.12.


静态类型
========

在传统上，在 C 代码中定义的类型都是 *静态的*，也就是说，"PyTypeObject"
结构体在代码中直接定义并使用 "PyType_Ready()" 来初始化。

这就导致了与在 Python 中定义的类型相关联的类型限制：

* 静态类型只能拥有一个基类；换句话说，他们不能使用多重继承。

* 静态类型对象（但并非它们的实例）是不可变对象。 不可能在 Python 中添
  加或修改类型对象的属性。

* 静态类型对象是跨 子解释器 共享的，因此它们不应包括任何子解释器专属的
  状态。

此外，由于 "PyTypeObject" 只是作为不透明结构的 受限 API 的一部分，因此
任何使用静态类型的扩展模块都必须针对特定的 Python 次版本进行编译。


堆类型
======

一种 静态类型的 替代物是 *堆分配类型*，或者简称 *堆类型*，它与使用
Python 的 "class" 语句创建的类紧密对应。 堆类型设置了
"Py_TPFLAGS_HEAPTYPE" 旗标。

这是通过填充 "PyType_Spec" 结构体并调用 "PyType_FromSpec()",
"PyType_FromSpecWithBases()", "PyType_FromModuleAndSpec()" 或
"PyType_FromMetaclass()" 来实现的。


数字对象结构体
**************

type PyNumberMethods

   该结构体持有指向被对象用来实现数字协议的函数的指针。 每个函数都被
   数字协议 一节中记录的对应名称的函数所使用。

   结构体定义如下:

      typedef struct {
           binaryfunc nb_add;
           binaryfunc nb_subtract;
           binaryfunc nb_multiply;
           binaryfunc nb_remainder;
           binaryfunc nb_divmod;
           ternaryfunc nb_power;
           unaryfunc nb_negative;
           unaryfunc nb_positive;
           unaryfunc nb_absolute;
           inquiry nb_bool;
           unaryfunc nb_invert;
           binaryfunc nb_lshift;
           binaryfunc nb_rshift;
           binaryfunc nb_and;
           binaryfunc nb_xor;
           binaryfunc nb_or;
           unaryfunc nb_int;
           void *nb_reserved;
           unaryfunc nb_float;

           binaryfunc nb_inplace_add;
           binaryfunc nb_inplace_subtract;
           binaryfunc nb_inplace_multiply;
           binaryfunc nb_inplace_remainder;
           ternaryfunc nb_inplace_power;
           binaryfunc nb_inplace_lshift;
           binaryfunc nb_inplace_rshift;
           binaryfunc nb_inplace_and;
           binaryfunc nb_inplace_xor;
           binaryfunc nb_inplace_or;

           binaryfunc nb_floor_divide;
           binaryfunc nb_true_divide;
           binaryfunc nb_inplace_floor_divide;
           binaryfunc nb_inplace_true_divide;

           unaryfunc nb_index;

           binaryfunc nb_matrix_multiply;
           binaryfunc nb_inplace_matrix_multiply;
      } PyNumberMethods;

   备注:

     双目和三目函数必须检查其所有操作数的类型，并实现必要的转换（至少
     有一个操作数是所定义类型的实例）。 如果没有为所给出的操作数定义操
     作，则双目和三目函数必须返回 "Py_NotImplemented"，如果发生了其他
     错误则它们必须返回 "NULL" 并设置一个异常。

   备注:

     "nb_reserved" 字段应当始终为 "NULL"。 在之前版本中其名称为
     "nb_long"，并在 Python 3.0.1 中改名。

binaryfunc PyNumberMethods.nb_add

binaryfunc PyNumberMethods.nb_subtract

binaryfunc PyNumberMethods.nb_multiply

binaryfunc PyNumberMethods.nb_remainder

binaryfunc PyNumberMethods.nb_divmod

ternaryfunc PyNumberMethods.nb_power

unaryfunc PyNumberMethods.nb_negative

unaryfunc PyNumberMethods.nb_positive

unaryfunc PyNumberMethods.nb_absolute

inquiry PyNumberMethods.nb_bool

unaryfunc PyNumberMethods.nb_invert

binaryfunc PyNumberMethods.nb_lshift

binaryfunc PyNumberMethods.nb_rshift

binaryfunc PyNumberMethods.nb_and

binaryfunc PyNumberMethods.nb_xor

binaryfunc PyNumberMethods.nb_or

unaryfunc PyNumberMethods.nb_int

void *PyNumberMethods.nb_reserved

unaryfunc PyNumberMethods.nb_float

binaryfunc PyNumberMethods.nb_inplace_add

binaryfunc PyNumberMethods.nb_inplace_subtract

binaryfunc PyNumberMethods.nb_inplace_multiply

binaryfunc PyNumberMethods.nb_inplace_remainder

ternaryfunc PyNumberMethods.nb_inplace_power

binaryfunc PyNumberMethods.nb_inplace_lshift

binaryfunc PyNumberMethods.nb_inplace_rshift

binaryfunc PyNumberMethods.nb_inplace_and

binaryfunc PyNumberMethods.nb_inplace_xor

binaryfunc PyNumberMethods.nb_inplace_or

binaryfunc PyNumberMethods.nb_floor_divide

binaryfunc PyNumberMethods.nb_true_divide

binaryfunc PyNumberMethods.nb_inplace_floor_divide

binaryfunc PyNumberMethods.nb_inplace_true_divide

unaryfunc PyNumberMethods.nb_index

binaryfunc PyNumberMethods.nb_matrix_multiply

binaryfunc PyNumberMethods.nb_inplace_matrix_multiply


映射对象结构体
**************

type PyMappingMethods

   该结构体持有指向对象用于实现映射协议的函数的指针。 它有三个成员：

lenfunc PyMappingMethods.mp_length

   该函数将被 "PyMapping_Size()" 和 "PyObject_Size()" 使用，并具有相同
   的签名。 如果对象没有定义长度则此槽位可被设为 "NULL"。

binaryfunc PyMappingMethods.mp_subscript

   该函数将被 "PyObject_GetItem()" 和 "PySequence_GetSlice()" 使用，并
   具有与 "PyObject_GetItem()" 相同的签名。 此槽位必须被填充以便
   "PyMapping_Check()" 函数返回 "1"，否则它可以为 "NULL"。

objobjargproc PyMappingMethods.mp_ass_subscript

   该函数将被 "PyObject_SetItem()", "PyObject_DelItem()",
   "PySequence_SetSlice()" 和 "PySequence_DelSlice()" 使用。 它具有与
   "PyObject_SetItem()" 相同的签名，但 *v* 也可以被设为 "NULL" 以删除
   一个条目。 如果此槽位为 "NULL"，则对象将不支持条目赋值和删除。


序列对象结构体
**************

type PySequenceMethods

   该结构体持有指向对象用于实现序列协议的函数的指针。

lenfunc PySequenceMethods.sq_length

   此函数被 "PySequence_Size()" 和 "PyObject_Size()" 所使用，并具有与
   它们相同的签名。 它还被用于通过 "sq_item" 和 "sq_ass_item" 槽位来处
   理负索引号。

binaryfunc PySequenceMethods.sq_concat

   此函数被 "PySequence_Concat()" 所使用并具有相同的签名。 在尝试通过
   "nb_add" 槽位执行数值相加之后它还会被用于 "+" 运算符。

ssizeargfunc PySequenceMethods.sq_repeat

   此函数被 "PySequence_Repeat()" 所使用并具有相同的签名。 在尝试通过
   "nb_multiply" 槽位执行数值相乘之后它还会被用于 "*" 运算符。

ssizeargfunc PySequenceMethods.sq_item

   此函数被 "PySequence_GetItem()" 所使用并具有相同的签名。 在尝试通过
   "mp_subscript" 槽位执行下标操作之后它还会被用于
   "PyObject_GetItem()"。 该槽位必须被填充以便 "PySequence_Check()" 函
   数返回 "1"，否则它可以为 "NULL"。

   负索引号是按如下方式处理的：如果 "sq_length" 槽位已被填充，它将被调
   用并使用序列长度来计算出正索引号并传给 "sq_item"。 如果 "sq_length"
   为 "NULL"，索引号将原样传给此函数。

ssizeobjargproc PySequenceMethods.sq_ass_item

   此函数被 "PySequence_SetItem()" 所使用并具有相同的签名。 在尝试通过
   "mp_ass_subscript" 槽位执行条目赋值和删除操作之后它还会被用于
   "PyObject_SetItem()" 和 "PyObject_DelItem()"。 如果对象不支持条目和
   删除则该槽位可以保持为 "NULL"。

objobjproc PySequenceMethods.sq_contains

   该函数可供 "PySequence_Contains()" 使用并具有相同的签名。 此槽位可
   以保持为 "NULL"，在此情况下 "PySequence_Contains()" 只需遍历该序列
   直到找到一个匹配。

binaryfunc PySequenceMethods.sq_inplace_concat

   此函数被 "PySequence_InPlaceConcat()" 所使用并具有相同的签名。 它应
   当修改它的第一个操作数，并将其返回。 该槽位可以保持为 "NULL"，在此
   情况下 "PySequence_InPlaceConcat()" 将回退到 "PySequence_Concat()"
   。 在尝试通过 "nb_inplace_add" 槽位执行数字原地相加之后它还会被用于
   增强赋值运算符 "+="。

ssizeargfunc PySequenceMethods.sq_inplace_repeat

   此函数被 "PySequence_InPlaceRepeat()" 所使用并具有相同的签名。 它应
   当修改它的第一个操作数，并将其返回。 该槽位可以保持为 "NULL"，在此
   情况下 "PySequence_InPlaceRepeat()" 将回退到 "PySequence_Repeat()"
   。 在尝试通过 "nb_inplace_multiply" 槽位执行数字原地相乘之后它还会
   被用于增强赋值运算符 "*="。


缓冲区对象结构体
****************

type PyBufferProcs

   此结构体持有指向 缓冲区协议 所需要的函数的指针。 该协议定义了导出方
   对象要如何向消费方对象暴露其内部数据。

getbufferproc PyBufferProcs.bf_getbuffer

   此函数的签名为:

      int (PyObject *exporter, Py_buffer *view, int flags);

   处理发给 *exporter* 的请求来填充 *flags* 所指定的 *view*。 除第 (3)
   点外，此函数的实现必须执行以下步骤：

   1. 检查请求是否能被满足。 如果不能，则会引发 "BufferError"，将
      view->obj 设为 "NULL" 并返回 "-1"。

   2. 填充请求的字段。

   3. 递增用于保存导出次数的内部计数器。

   4. 将 view->obj 设为 *exporter* 并递增 view->obj。

   5. 返回 "0"。

   如果 *exporter* 是缓冲区提供方的链式或树型结构的一部分，则可以使用
   两种主要方案：

   * 重导出：树型结构的每个成员作为导出对象并将 view->obj 设为对其自身
     的新引用。

   * 重定向：缓冲区请求将被重定向到树型结构的根对象。 在此，view->obj
     将为对根对象的新引用。

   *view* 中每个字段的描述参见 缓冲区结构体 一节，导出方对于特定请求应
   当如何反应参见 缓冲区请求类型 一节。

   所有在 "Py_buffer" 结构体中被指向的内存都属于导出方并必须保持有效直
   到不再有任何消费方。 "format", "shape", "strides", "suboffsets" 和
   "internal" 对于消费方来说是只读的。

   "PyBuffer_FillInfo()" 提供了一种暴露简单字节缓冲区同时正确处理地所
   有请求类型的简便方式。

   "PyObject_GetBuffer()" 是针对包装此函数的消费方的接口。

releasebufferproc PyBufferProcs.bf_releasebuffer

   此函数的签名为:

      void (PyObject *exporter, Py_buffer *view);

   处理释放缓冲区资源的请求。 如果不需要释放任何资源，则
   "PyBufferProcs.bf_releasebuffer" 可以为 "NULL"。 在其他情况下，此函
   数的标准实现将执行以下的可选步骤：

   1. 递减用于保存导出次数的内部计数器。

   2. 如果计数器为 "0"，则释放所有关联到 *view* 的内存。

   导出方必须使用 "internal" 字段来记录缓冲区专属的资源。 该字段将确保
   恒定，而消费方则可能将原始缓冲区作为 *view* 参数传入。

   此函数不可递减 view->obj，因为这是在 "PyBuffer_Release()" 中自动完
   成的（此方案适用于打破循环引用）。

   "PyBuffer_Release()" 是针对包装此函数的消费方的接口。


异步对象结构体
**************

Added in version 3.5.

type PyAsyncMethods

   此结构体将持有指向需要用来实现 *awaitable* 和 *asynchronous
   iterator* 对象的函数的指针。

   结构体定义如下:

      typedef struct {
          unaryfunc am_await;
          unaryfunc am_aiter;
          unaryfunc am_anext;
          sendfunc am_send;
      } PyAsyncMethods;

unaryfunc PyAsyncMethods.am_await

   此函数的签名为:

      PyObject *am_await(PyObject *self);

   返回的对象必须为 *iterator*，即对其执行 "PyIter_Check()" 必须返回
   "1"。

   如果一个对象不是 *awaitable* 则此槽位可被设为 "NULL"。

unaryfunc PyAsyncMethods.am_aiter

   此函数的签名为:

      PyObject *am_aiter(PyObject *self);

   必须返回一个 *asynchronous iterator* 对象。 请参阅 "__anext__()" 了
   解详情。

   如果一个对象没有实现异步迭代协议则此槽位可被设为 "NULL"。

unaryfunc PyAsyncMethods.am_anext

   此函数的签名为:

      PyObject *am_anext(PyObject *self);

   必须返回一个 *awaitable* 对象。 请参阅 "__anext__()" 了解详情。 此
   槽位可被设为 "NULL"。

sendfunc PyAsyncMethods.am_send

   此函数的签名为:

      PySendResult am_send(PyObject *self, PyObject *arg, PyObject **result);

   请参阅 "PyIter_Send()" 了解详情。 此槽位可被设为 "NULL"。

   Added in version 3.10.


槽位类型 typedef
****************

typedef PyObject *(*allocfunc)(PyTypeObject *cls, Py_ssize_t nitems)
    * 属于 稳定 ABI.*

   此函数的设计目标是将内存分配与内存初始化进行分离。 它应当返回一个指
   向足够容纳实例长度，适当对齐，并初始化为零的内存块的指针，但将
   "ob_refcnt" 设为 "1" 并将 "ob_type" 设为 type 参数。 如果类型的
   "tp_itemsize" 为非零值，则对象的 "ob_size" 字段应当被初始化为
   *nitems* 而分配内存块的长度应为 "tp_basicsize + nitems*tp_itemsize"
   ，并舍入到 "sizeof(void*)" 的倍数；在其他情况下，*nitems* 将不会被
   使用而内存块的长度应为 "tp_basicsize"。

   此函数不应执行任何其他实例初始化操作，即使是分配额外内存也不应执行
   ；那应当由 "tp_new" 来完成。

typedef void (*destructor)(PyObject*)
    * 属于 稳定 ABI.*

typedef void (*freefunc)(void*)

   参见 "tp_free"。

typedef PyObject *(*newfunc)(PyObject*, PyObject*, PyObject*)
    * 属于 稳定 ABI.*

   参见 "tp_new"。

typedef int (*initproc)(PyObject*, PyObject*, PyObject*)
    * 属于 稳定 ABI.*

   参见 "tp_init"。

typedef PyObject *(*reprfunc)(PyObject*)
    * 属于 稳定 ABI.*

   参见 "tp_repr"。

typedef PyObject *(*getattrfunc)(PyObject *self, char *attr)
    * 属于 稳定 ABI.*

   返回对象的指定属性的值。

typedef int (*setattrfunc)(PyObject *self, char *attr, PyObject *value)
    * 属于 稳定 ABI.*

   为对象设置指定属性的值。 将 value 参数设为 "NULL" 将删除该属性。

typedef PyObject *(*getattrofunc)(PyObject *self, PyObject *attr)
    * 属于 稳定 ABI.*

   返回对象的指定属性的值。

   参见 "tp_getattro"。

typedef int (*setattrofunc)(PyObject *self, PyObject *attr, PyObject *value)
    * 属于 稳定 ABI.*

   为对象设置指定属性的值。 将 value 参数设为 "NULL" 将删除该属性。

   参见 "tp_setattro"。

typedef PyObject *(*descrgetfunc)(PyObject*, PyObject*, PyObject*)
    * 属于 稳定 ABI.*

   参见 "tp_descr_get"。

typedef int (*descrsetfunc)(PyObject*, PyObject*, PyObject*)
    * 属于 稳定 ABI.*

   参见 "tp_descr_set"。

typedef Py_hash_t (*hashfunc)(PyObject*)
    * 属于 稳定 ABI.*

   参见 "tp_hash"。

typedef PyObject *(*richcmpfunc)(PyObject*, PyObject*, int)
    * 属于 稳定 ABI.*

   参见 "tp_richcompare"。

typedef PyObject *(*getiterfunc)(PyObject*)
    * 属于 稳定 ABI.*

   参见 "tp_iter"。

typedef PyObject *(*iternextfunc)(PyObject*)
    * 属于 稳定 ABI.*

   参见 "tp_iternext"。

typedef Py_ssize_t (*lenfunc)(PyObject*)
    * 属于 稳定 ABI.*

typedef int (*getbufferproc)(PyObject*, Py_buffer*, int)
    * 属于 稳定 ABI 自 3.12 版起.*

typedef void (*releasebufferproc)(PyObject*, Py_buffer*)
    * 属于 稳定 ABI 自 3.12 版起.*

typedef PyObject *(*unaryfunc)(PyObject*)
    * 属于 稳定 ABI.*

typedef PyObject *(*binaryfunc)(PyObject*, PyObject*)
    * 属于 稳定 ABI.*

typedef PySendResult (*sendfunc)(PyObject*, PyObject*, PyObject**)

   参见 "am_send"。

typedef PyObject *(*ternaryfunc)(PyObject*, PyObject*, PyObject*)
    * 属于 稳定 ABI.*

typedef PyObject *(*ssizeargfunc)(PyObject*, Py_ssize_t)
    * 属于 稳定 ABI.*

typedef int (*ssizeobjargproc)(PyObject*, Py_ssize_t, PyObject*)
    * 属于 稳定 ABI.*

typedef int (*objobjproc)(PyObject*, PyObject*)
    * 属于 稳定 ABI.*

typedef int (*objobjargproc)(PyObject*, PyObject*, PyObject*)
    * 属于 稳定 ABI.*


例子
****

下面是一些 Python 类型定义的简单示例。 其中包括你可能会遇到的通常用法
。 有些演示了令人困惑的边际情况。 要获取更多示例、实践信息以及教程，请
参阅 自定义扩展类型：教程 和 定义扩展类型：已分类主题。

一个基本的 静态类型:

   typedef struct {
       PyObject_HEAD
       const char *data;
   } MyObject;

   static PyTypeObject MyObject_Type = {
       PyVarObject_HEAD_INIT(NULL, 0)
       .tp_name = "mymod.MyObject",
       .tp_basicsize = sizeof(MyObject),
       .tp_doc = PyDoc_STR("My objects"),
       .tp_new = myobj_new,
       .tp_dealloc = (destructor)myobj_dealloc,
       .tp_repr = (reprfunc)myobj_repr,
   };

你可能还会看到带有更繁琐的初始化器的较旧代码（特别是在 CPython 代码库
中）:

   static PyTypeObject MyObject_Type = {
       PyVarObject_HEAD_INIT(NULL, 0)
       "mymod.MyObject",               /* tp_name */
       sizeof(MyObject),               /* tp_basicsize */
       0,                              /* tp_itemsize */
       (destructor)myobj_dealloc,      /* tp_dealloc */
       0,                              /* tp_vectorcall_offset */
       0,                              /* tp_getattr */
       0,                              /* tp_setattr */
       0,                              /* tp_as_async */
       (reprfunc)myobj_repr,           /* tp_repr */
       0,                              /* tp_as_number */
       0,                              /* tp_as_sequence */
       0,                              /* tp_as_mapping */
       0,                              /* tp_hash */
       0,                              /* tp_call */
       0,                              /* tp_str */
       0,                              /* tp_getattro */
       0,                              /* tp_setattro */
       0,                              /* tp_as_buffer */
       0,                              /* tp_flags */
       PyDoc_STR("My objects"),        /* tp_doc */
       0,                              /* tp_traverse */
       0,                              /* tp_clear */
       0,                              /* tp_richcompare */
       0,                              /* tp_weaklistoffset */
       0,                              /* tp_iter */
       0,                              /* tp_iternext */
       0,                              /* tp_methods */
       0,                              /* tp_members */
       0,                              /* tp_getset */
       0,                              /* tp_base */
       0,                              /* tp_dict */
       0,                              /* tp_descr_get */
       0,                              /* tp_descr_set */
       0,                              /* tp_dictoffset */
       0,                              /* tp_init */
       0,                              /* tp_alloc */
       myobj_new,                      /* tp_new */
   };

一个支持弱引用、实例字典和哈希运算的类型:

   typedef struct {
       PyObject_HEAD
       const char *data;
   } MyObject;

   static PyTypeObject MyObject_Type = {
       PyVarObject_HEAD_INIT(NULL, 0)
       .tp_name = "mymod.MyObject",
       .tp_basicsize = sizeof(MyObject),
       .tp_doc = PyDoc_STR("My objects"),
       .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE |
            Py_TPFLAGS_HAVE_GC | Py_TPFLAGS_MANAGED_DICT |
            Py_TPFLAGS_MANAGED_WEAKREF,
       .tp_new = myobj_new,
       .tp_traverse = (traverseproc)myobj_traverse,
       .tp_clear = (inquiry)myobj_clear,
       .tp_alloc = PyType_GenericNew,
       .tp_dealloc = (destructor)myobj_dealloc,
       .tp_repr = (reprfunc)myobj_repr,
       .tp_hash = (hashfunc)myobj_hash,
       .tp_richcompare = PyBaseObject_Type.tp_richcompare,
   };

一个不能被子类化且不能被调用以使用 "Py_TPFLAGS_DISALLOW_INSTANTIATION"
旗标创建实例（例如使用单独的工厂函数）的 str 子类:

   typedef struct {
       PyUnicodeObject raw;
       char *extra;
   } MyStr;

   static PyTypeObject MyStr_Type = {
       PyVarObject_HEAD_INIT(NULL, 0)
       .tp_name = "mymod.MyStr",
       .tp_basicsize = sizeof(MyStr),
       .tp_base = NULL,  // set to &PyUnicode_Type in module init
       .tp_doc = PyDoc_STR("my custom str"),
       .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_DISALLOW_INSTANTIATION,
       .tp_repr = (reprfunc)myobj_repr,
   };

最简单的固定长度实例 静态类型:

   typedef struct {
       PyObject_HEAD
   } MyObject;

   static PyTypeObject MyObject_Type = {
       PyVarObject_HEAD_INIT(NULL, 0)
       .tp_name = "mymod.MyObject",
   };

最简单的具有可变长度实例的 静态类型:

   typedef struct {
       PyObject_VAR_HEAD
       const char *data[1];
   } MyObject;

   static PyTypeObject MyObject_Type = {
       PyVarObject_HEAD_INIT(NULL, 0)
       .tp_name = "mymod.MyObject",
       .tp_basicsize = sizeof(MyObject) - sizeof(char *),
       .tp_itemsize = sizeof(char *),
   };
