Python对象体系深入分析


Posted in Python onOctober 28, 2014

本文较为详细的分析了了Python的对象体系。分享给大家供大家参考。具体如下:

Guido用C语言创造了Python,在Python的世界中一切皆为对象.

一.C视角中的Python对象

让我们一起追溯到源头,Python由C语言实现,且向外提供了C的API http://docs.python.org/c-api/index.html .

我们思考问题的时候,可能对于对象这种东西很容易理解,而计算机能理解的只有0,1序列这样的字节序列,从根本上讲,我们所说的计算机语言中的对象只是在内存中的一块内存空间里的0,1序列而已,这些连续或者非连续的内存空间在更高层次上可以看作是一个整体.在Python中,我们所提到的一般的对象都是C中的结构体在堆Heap上申请的一块内存空间.

为了能够用C语言实现Python的面向对象的机制,需要定义一些结构体,能够操作那些对象的内存空间。

1.PyObject&PyVarObject

所有的Python对象都有一些共同的东西,我们将其高度抽象成一个结构体PyObject

typedef struct _object{  

    PyObject_HEAD  

} PyObject;  

//其实PyObject_HEAD这个宏在发行版本中的为  

int ob_refcnt;  

struct _typeobject *ob_type;

ob_refcnt,就是对象引用计数,它的存在是为了实现了Python的基于引用技术的垃圾回收机制.

还有一个是指向一个类型对象结构体的指针,用以代表该对象的类型.

在C语言的实现的时候,还有一个结构体扩展于PyObject

那便是PyVarObject,其内容为PyObject_VAR_HEAD这个宏,它比PyObject多了一个ob_size,用来表示变长对象的长度,详情见http://docs.python.org/c-api/structures.html

还有一点请大家不要搞混,这里的PyObject和PyVarObject和Python世界中的真实对象没有对应关系,这两个只是Python对象全体在C语言表示中的一种抽象.也就是说在C语言中,只要是一个Python对象结构体的数据,那么其内存的开始部分都会有上面结构体的几个变量,所以一个PyObject的指针便可以指向所有的C语言中的表示Python对象的结构体,这样在C语言的实现中,我们便可以通过这个统一的指针操作所有的内置的Python对象结构体了.

2.PyTypeObject

刚刚还有一个东西没有讲,那便是_typeobject(PyTypeObject)这个结构体,它是Python中所有类型对象的抽象,这样我们在C语言的层次里对于所有的类型对象结构体都可以通过PyTypeObject的指针来调用

typedef struct _typeobject {  

//注意开始部分为PyObject_VAR_HEAD  

PyObject_VAR_HEAD  

char *tp_name; /* For printing, in format  

"<module>.<name>" */  

int tp_basicsize, tp_itemsize; /* For allocation */  

/* Methods to implement standard operations */  

destructor tp_dealloc;  

printfunc tp_print;  

……  

/* More standard operations (here for  

binary compatibility) */  

hashfunc tp_hash;  

ternaryfunc tp_call;  

……  

} PyTypeObject;

3.Python内置对象和C结构体的对应

现在Python面向对象机制的对象和类型的抽象都已经说过了,接下来我们来看下在python中真实存在的对象在C语言实现的时候是怎么样的呢?

首先需要谈的是那些Python的内置对象,这些都是C语言定义了的,当Python环境初始化后,这些对象便创建好了。

PyAPI_DATA(PyTypeObject) PyType_Type; /* built-in 'type' */  

PyAPI_DATA(PyTypeObject) PyBaseObject_Type; /* built-in 'object' */

object对象在Python中是一个比较基础的对象,它在C语言中对应的结构体是PyBaseObject_Type,从C语言中的这个命名我们可以大概知道这个类是一个类型对象.

还有就是Python中的<type 'type'>在C语言中对应着PyType_Type

PyTypeObject PyType_Type = {  

PyObject_HEAD_INIT(&PyType_Type)  

0, /* ob_size */  

"type", /* tp_name */  

sizeof(PyHeapTypeObject), /* tp_basicsize */  

sizeof(PyMemberDef), /* tp_itemsize */  

……  

};

我们再看看比较具体的整数

一个整数instance在C语言中的表示的结构体是PyIntObject

typedef struct {  

PyObject_HEAD  

long ob_ival;  

} PyIntObject;
 

也就是说通过这样的结构体我们就可以在C语言的的运行时中指向Python的整数对象.

那么相应的我们Python的整数类型对象为

yTypeObject PyInt_Type = {  

PyObject_HEAD_INIT(&PyType_Type)  

0,  

"int",  

sizeof(PyIntObject),  

……  

};

4.自定义对象

当我们创建一个Python对象的时候,最终都是通过Python的底层来做的,

当我们通过Python语言定义了自己的一个class A之后,Python首先根据你写的代码创建了一个A这样的class对象(类对象),然后当你需要创建A的实例的时候,其实在Python的底层都是通过A这个Class对象进行创建的。

二.Python视角中的对象体系

在单纯的Python的世界中,一切都是对象.这些对象可以分为三类,

metaclasses,classes,instance

其中classes又可以分为内置的type和用户自定义的class

下面我们通过一张图片来作详细的说明

Python对象体系深入分析

注:

其中C的定义的方式如下(python 中继承于某类直接写在类名后面的括号中):

class C(object):  

   ......

其中实线表示 is-kind-of 的关系 ,虚线表示is-instance-of的关系.

查看当前classes对象(instances对象没有__bases__属性)的基类的时候,可以用过classes_name.__bases__进行查看,其值为一个Tuple元组(Python支持多继承).

查看当前对象的类型的方法是object_name.__class__

我们可以通过一些测试来证实上面的图:

Python对象体系深入分析

type为所有类的类。

希望本文所述对大家的Python程序设计有所帮助。

Python 相关文章推荐
基于Django模板中的数字自增(详解)
Sep 05 Python
利用python为运维人员写一个监控脚本
Mar 25 Python
python3去掉string中的标点符号方法
Jan 22 Python
Pandas统计重复的列里面的值方法
Jan 30 Python
Pandas库之DataFrame使用的学习笔记
Jun 21 Python
flask应用部署到服务器的方法
Jul 12 Python
详解centos7+django+python3+mysql+阿里云部署项目全流程
Nov 15 Python
python 统计文件中的字符串数目示例
Dec 24 Python
MNIST数据集转化为二维图片的实现示例
Jan 10 Python
关于Python字符串显示u...的解决方式
Mar 06 Python
selenium+超级鹰实现模拟登录12306
Jan 24 Python
如何用python绘制雷达图
Apr 24 Python
Python中类的继承代码实例
Oct 28 #Python
Python列表list数组array用法实例解析
Oct 28 #Python
python实现无证书加密解密实例
Oct 27 #Python
深入理解Python 代码优化详解
Oct 27 #Python
简单的Python抓taobao图片爬虫
Oct 26 #Python
Python性能优化的20条建议
Oct 25 #Python
跟老齐学Python之网站的结构
Oct 24 #Python
You might like
PHP入门教程之上传文件实例详解
2016/09/11 PHP
php获取excel文件数据
2017/04/21 PHP
JQuery扩展插件Validate 3通过参数设置错误信息
2011/09/05 Javascript
js中prototype用法详细介绍
2013/11/14 Javascript
深入了解JavaScript中的Symbol的使用方法
2015/07/28 Javascript
Javascript数组Array基础介绍
2016/03/13 Javascript
Javascript 引擎工作机制详解
2016/11/30 Javascript
layui radio性别单选框赋值方法
2018/08/15 Javascript
分享5个小技巧让你写出更好的 JavaScript 条件语句
2018/10/20 Javascript
浅谈小程序 setData学问多
2019/02/20 Javascript
Vue使用zTree插件封装树组件操作示例
2019/04/25 Javascript
vue+ts下对axios的封装实现
2020/02/18 Javascript
[03:52]显微镜下的DOTA2第三期——英雄在无聊的时候干什么
2014/06/20 DOTA
[06:40]2014DOTA2西雅图国际邀请赛 DK战队巡礼
2014/07/07 DOTA
[02:29]完美世界高校联赛上海赛区回顾
2015/12/15 DOTA
Python正则表达式介绍
2012/08/06 Python
python共享引用(多个变量引用)示例代码
2013/12/04 Python
详解Python中最难理解的点-装饰器
2017/04/03 Python
Python学习小技巧之列表项的排序
2017/05/20 Python
Python 2.7中文显示与处理方法
2018/07/16 Python
Python3 Tkinter选择路径功能的实现方法
2019/06/14 Python
HTML5中的websocket实现直播功能
2018/05/21 HTML / CSS
美国知名艺术画网站:Art.com
2017/02/09 全球购物
Grow Gorgeous美国官网:只要八天,体验唤醒毛囊后新生的茂密秀发
2018/06/04 全球购物
英国玛莎百货新西兰:Marks & Spencer New Zealand
2019/07/21 全球购物
三星法国官方网站:Samsung法国
2019/10/31 全球购物
美国踏板车和轻便摩托车销售网站:Mega Motor Madness
2020/02/26 全球购物
写出SQL四条最基本的数据操作语句(DML)
2012/12/12 面试题
Laravel的加密解密与哈希实例讲解
2021/03/24 PHP
期末考试动员演讲稿
2014/01/10 职场文书
大学生全国两会报告感想
2014/03/17 职场文书
客运企业隐患排查工作方案
2014/06/06 职场文书
违纪检讨书
2015/01/27 职场文书
社区植树节活动总结
2015/02/06 职场文书
卫生保健工作总结2015
2015/05/18 职场文书
《金肉人》米特&《航海王》阿鹤声优松岛实因胰脏癌去世 享寿81岁
2022/04/13 日漫