深入源码解析Python中的对象与类型


Posted in Python onDecember 11, 2015

对象
对象, 在C语言是如何实现的?

Python中对象分为两类: 定长(int等), 非定长(list/dict等)

所有对象都有一些相同的东西, 源码中定义为PyObject和PyVarObject, 两个定义都有一个共同的头部定义PyObject_HEAD(其实PyVarObject有自己的头部定义PyObject_VAR_HEAD, 但其实际上用的也是PyObject_HEAD).

源码位置: Include/object.h

PyObject_HEAD
Python 内部, 每个对象拥有相同的头部.

定义

/* PyObject_HEAD defines the initial segment of every PyObject. */
#define PyObject_HEAD          \
  _PyObject_HEAD_EXTRA        \
  Py_ssize_t ob_refcnt;        \
  struct _typeobject *ob_type;

说明

1. _PyObject_HEAD_EXTRA
先忽略, 双向链表结构, 后面垃圾回收再说

2. Py_ssize_t ob_refcnt
Py_ssize_t在编译时确定, 整型
ob_refcnt, 引用计数, 跟Python的内存管理机制相关(基于引用计数的垃圾回收)

3. struct _typeobject *ob_type
*ob_type 指向类型对象的指针(指向_typeobject结构体)
决定了这个对象的类型!
PyObject
定义

typedef struct _object {
   PyObject_HEAD
 } PyObject;

说明

 1. 依赖关系
 PyObject -> PyObject_HEAD
结构

深入源码解析Python中的对象与类型

PyVarObject
定义

typedef struct {
  PyObject_VAR_HEAD
} PyVarObject;

#define PyObject_VAR_HEAD        \
 PyObject_HEAD            \
 Py_ssize_t ob_size; /* Number of items in variable part */

说明

 1. 依赖关系
 PyVarObject -> PyObject_VAR_HEAD -> PyObject_HEAD

 2.Py_ssize_t ob_size
 ob_size, 变长对象容纳的元素个数
结构

深入源码解析Python中的对象与类型

代码关系

深入源码解析Python中的对象与类型

几个方法
跟对象相关的方法

#define Py_REFCNT(ob)           (((PyObject*)(ob))->ob_refcnt)
读取引用计数

#define Py_TYPE(ob)             (((PyObject*)(ob))->ob_type)
获取对象类型

#define Py_SIZE(ob)             (((PyVarObject*)(ob))->ob_size)
读取元素个数(len)
跟引用计数相关的方法

Py_INCREF(op)  增加对象引用计数

Py_DECREF(op)  减少对象引用计数, 如果计数位0, 调用_Py_Dealloc

_Py_Dealloc(op) 调用对应类型的 tp_dealloc 方法(每种类型回收行为不一样的, 各种缓存池机制, 后面看)
其他
几个参数涉及

ob_refcnt 引用计数, 与内存管理/垃圾回收相关
ob_type   类型, 涉及Python的类型系统

类型
一个例子

>>> a = 1
>>> a
1

>>> type(a)
<type 'int'>

#等价的两个
>>> type(type(a))
<type 'type'>
>>> type(int)
<type 'type'>

#还是等价的两个
>>> type(type(type(a)))
<type 'type'>
>>> type(type(int))
<type 'type'>

我们反向推导一个int对象是怎么生成的.

1. 首先, 定义一种类型叫PyTypeObject
代码位置 Include/object.h

定义

 

typedef struct _typeobject {

 /* MARK: base, 注意, 是个变长对象*/
 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 */
 printfunc tp_print;
 hashfunc tp_hash;

 /* Method suites for standard classes */
 PyNumberMethods *tp_as_number;  // 数值对象操作
 PySequenceMethods *tp_as_sequence; // 序列对象操作
 PyMappingMethods *tp_as_mapping; // 字典对象操作

 // 一堆属性定义
 ....

} PyTypeObject;

说明

1. PyObject_VAR_HEAD
变长对象

2. const char *tp_name
tp_name, 类型名字符串数组
所有Type都是PyTypeObject的"实例": PyType_Type/PyInt_Type

2. 然后, 用PyTypeObject初始化得到一个对象PyType_Type
代码位置 Objects/typeobject.c

定义

PyTypeObject PyType_Type = {
 PyVarObject_HEAD_INIT(&PyType_Type, 0)
 "type",                   /* tp_name */
 sizeof(PyHeapTypeObject),          /* tp_basicsize */
 sizeof(PyMemberDef),            /* tp_itemsize */
 (destructor)type_dealloc,          /* tp_dealloc */

 // type对象的方法和属性初始化值
 .....

};

说明

1. tp_name
类型名, 这里是"type"

2. PyVarObject_HEAD_INIT(&PyType_Type, 0)
PyVarObject_HEAD_INIT, 这个方法在 Include/object.h中,
等价于
        ob_refcnt = 1
        *ob_type = &PyType_Type
        ob_size = 0

即, PyType_Type的类型是其本身!
结构

第一张图, 箭头表示实例化(google doc用不是很熟找不到对应类型的箭头)

深入源码解析Python中的对象与类型

第二张图, 箭头表示指向

深入源码解析Python中的对象与类型

使用

# 1. int 的 类型 是`type`
>>> type(int)
<type 'type'>

# 2. type 的类型 还是`type`, 对应上面说明第二点
>>> type(type(int))
<type 'type'>

注意: 无论任何时候, ob_type指向的是 PyTypeObject的实例: PyType_Type/PyInt_Type...

3. 再然后, 定义具体的类型, 这里以PyInt_Type为例子
代码位置 Objects/intobject.c

定义

PyTypeObject PyInt_Type = {
 PyVarObject_HEAD_INIT(&PyType_Type, 0)
 "int",
 sizeof(PyIntObject),
 0,

 // int类型的相关方法和属性值
 ....

 (hashfunc)int_hash,             /* tp_hash */

};

说明

1. "int"
PyInt_Type的类型名是int

2.PyVarObject_HEAD_INIT(&PyType_Type, 0)
PyInt_Type的

 

*ob_type = &PyType_Type

结构

深入源码解析Python中的对象与类型

使用

>>> type(1)
<type 'int'>

>>> type(type(1))
<type 'type'>

4. 最后, 生成一个整数对象int
代码位置 Include/intobject.h

定义

typedef struct {
  PyObject_HEAD
  long ob_ival;
} PyIntObject;

结构

深入源码解析Python中的对象与类型

Python 相关文章推荐
使用Python脚本在Linux下实现部分Bash Shell的教程
Apr 17 Python
Python中Random和Math模块学习笔记
May 18 Python
Python实现简单的代理服务器
Jul 25 Python
Python 网页解析HTMLParse的实例详解
Aug 10 Python
在cmd命令行里进入和退出Python程序的方法
May 12 Python
Python神奇的内置函数locals的实例讲解
Feb 22 Python
python3的url编码和解码,自定义gbk、utf-8的例子
Aug 22 Python
Python3 pandas 操作列表实例详解
Sep 23 Python
将python2.7添加进64位系统的注册表方式
Nov 20 Python
如何将你的应用迁移到Python3的三个步骤
Dec 22 Python
matplotlib绘制正余弦曲线图的实现
Feb 22 Python
python迷宫问题深度优先遍历实例
Jun 20 Python
Python实现各种排序算法的代码示例总结
Dec 11 #Python
Python操作MySQL数据库9个实用实例
Dec 11 #Python
使用Python编写简单的画图板程序的示例教程
Dec 08 #Python
一波神奇的Python语句、函数与方法的使用技巧总结
Dec 08 #Python
Python使用pygame模块编写俄罗斯方块游戏的代码实例
Dec 08 #Python
用Python抢过年的火车票附源码
Dec 07 #Python
Python随手笔记之标准类型内建函数
Dec 02 #Python
You might like
Server.HTMLEncode让代码在页面里显示为源代码
2013/12/08 PHP
php+ajax简单实现全选删除的方法
2016/12/06 PHP
PHP defined()函数的使用图文详解
2019/07/20 PHP
用js实现计算代码行数的简单方法附代码
2007/08/13 Javascript
我遇到的参数传递中 双引号单引号嵌套问题
2010/02/11 Javascript
扩展JavaScript功能的正确方法(译文)
2012/04/12 Javascript
javascript弹出层输入框(示例代码)
2013/12/11 Javascript
解决jquery中美元符号命名冲突问题
2014/01/08 Javascript
深入理解JavaScript系列(42):设计模式之原型模式详解
2015/03/04 Javascript
JavaScript实现带标题的图片轮播特效
2015/05/20 Javascript
jQuery实现图片向左向右切换效果的简单实例
2016/05/18 Javascript
plupload+artdialog实现多平台上传文件
2016/07/19 Javascript
微信小程序 自己制作小组件实例详解
2016/12/22 Javascript
vue-router路由与页面间导航实例解析
2017/11/07 Javascript
浅谈AngularJs 双向绑定原理(数据绑定机制)
2017/12/07 Javascript
微信小程序定位当前城市的方法
2018/07/19 Javascript
Python线程的两种编程方式
2015/04/14 Python
python处理multipart/form-data的请求方法
2018/12/26 Python
关于python中密码加盐的学习体会小结
2019/07/15 Python
Python爬取视频(其实是一篇福利)过程解析
2019/08/01 Python
flask框架json数据的拿取和返回操作示例
2019/11/28 Python
Selenium基于PIL实现拼接滚动截图
2020/04/10 Python
python 绘制场景热力图的示例
2020/09/23 Python
Marc O’Polo俄罗斯官方在线商店:德国高端时尚品牌
2019/12/26 全球购物
我们没有写servlet的构造方法,那么容器是怎么创建servlet的实例呢
2013/04/24 面试题
简历中求职的个人自我评价
2013/12/03 职场文书
会计学自荐信
2014/06/03 职场文书
北京奥运会口号
2014/06/21 职场文书
公司踏青活动方案
2014/08/16 职场文书
物价局领导班子四风问题整改措施
2014/10/26 职场文书
师德标兵事迹材料
2014/12/19 职场文书
2015中秋节慰问信范文
2015/03/23 职场文书
教导处教学工作总结
2015/08/12 职场文书
Pytorch distributed 多卡并行载入模型操作
2021/06/05 Python
漫画「你在春天醒来」第10卷封面公开
2022/03/21 日漫
PyTorch device与cuda.device用法
2022/04/03 Python