深入源碼解析Python中的對(duì)象與類型
對(duì)象
對(duì)象, 在C語言是如何實(shí)現(xiàn)的?
Python中對(duì)象分為兩類: 定長(zhǎng)(int等), 非定長(zhǎng)(list/dict等)
所有對(duì)象都有一些相同的東西, 源碼中定義為PyObject和PyVarObject, 兩個(gè)定義都有一個(gè)共同的頭部定義PyObject_HEAD(其實(shí)PyVarObject有自己的頭部定義PyObject_VAR_HEAD, 但其實(shí)際上用的也是PyObject_HEAD).
源碼位置: Include/object.h
PyObject_HEAD
Python 內(nèi)部, 每個(gè)對(duì)象擁有相同的頭部.
定義
/* 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
先忽略, 雙向鏈表結(jié)構(gòu), 后面垃圾回收再說
2. Py_ssize_t ob_refcnt
Py_ssize_t在編譯時(shí)確定, 整型
ob_refcnt, 引用計(jì)數(shù), 跟Python的內(nèi)存管理機(jī)制相關(guān)(基于引用計(jì)數(shù)的垃圾回收)
3. struct _typeobject *ob_type
*ob_type 指向類型對(duì)象的指針(指向_typeobject結(jié)構(gòu)體)
決定了這個(gè)對(duì)象的類型!
PyObject
定義
typedef struct _object { PyObject_HEAD } PyObject;
說明
1. 依賴關(guān)系
PyObject -> PyObject_HEAD
結(jié)構(gòu)
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. 依賴關(guān)系
PyVarObject -> PyObject_VAR_HEAD -> PyObject_HEAD
2.Py_ssize_t ob_size
ob_size, 變長(zhǎng)對(duì)象容納的元素個(gè)數(shù)
結(jié)構(gòu)
代碼關(guān)系
幾個(gè)方法
跟對(duì)象相關(guān)的方法
#define Py_REFCNT(ob) (((PyObject*)(ob))->ob_refcnt)
讀取引用計(jì)數(shù)
#define Py_TYPE(ob) (((PyObject*)(ob))->ob_type)
獲取對(duì)象類型
#define Py_SIZE(ob) (((PyVarObject*)(ob))->ob_size)
讀取元素個(gè)數(shù)(len)
跟引用計(jì)數(shù)相關(guān)的方法
Py_INCREF(op) 增加對(duì)象引用計(jì)數(shù)
Py_DECREF(op) 減少對(duì)象引用計(jì)數(shù), 如果計(jì)數(shù)位0, 調(diào)用_Py_Dealloc
_Py_Dealloc(op) 調(diào)用對(duì)應(yīng)類型的 tp_dealloc 方法(每種類型回收行為不一樣的, 各種緩存池機(jī)制, 后面看)
其他
幾個(gè)參數(shù)涉及
ob_refcnt 引用計(jì)數(shù), 與內(nèi)存管理/垃圾回收相關(guān)
ob_type 類型, 涉及Python的類型系統(tǒng)
類型
一個(gè)例子
>>> a = 1 >>> a 1 >>> type(a) <type 'int'> #等價(jià)的兩個(gè) >>> type(type(a)) <type 'type'> >>> type(int) <type 'type'> #還是等價(jià)的兩個(gè) >>> type(type(type(a))) <type 'type'> >>> type(type(int)) <type 'type'>
我們反向推導(dǎo)一個(gè)int對(duì)象是怎么生成的.
1. 首先, 定義一種類型叫PyTypeObject
代碼位置 Include/object.h
定義
typedef struct _typeobject { /* MARK: base, 注意, 是個(gè)變長(zhǎng)對(duì)象*/ PyObject_VAR_HEAD const char *tp_name; /* For printing, in format "<module>.<name>" */ //類型名 Py_ssize_t tp_basicsize, tp_itemsize; /* For allocation */ // 創(chuàng)建該類型對(duì)象時(shí)分配的內(nèi)存空間大小 // 一堆方法定義, 函數(shù)和指針 /* Methods to implement standard operations */ printfunc tp_print; hashfunc tp_hash; /* Method suites for standard classes */ PyNumberMethods *tp_as_number; // 數(shù)值對(duì)象操作 PySequenceMethods *tp_as_sequence; // 序列對(duì)象操作 PyMappingMethods *tp_as_mapping; // 字典對(duì)象操作 // 一堆屬性定義 .... } PyTypeObject;
說明
1. PyObject_VAR_HEAD
變長(zhǎng)對(duì)象
2. const char *tp_name
tp_name, 類型名字符串?dāng)?shù)組
所有Type都是PyTypeObject的"實(shí)例": PyType_Type/PyInt_Type
2. 然后, 用PyTypeObject初始化得到一個(gè)對(duì)象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對(duì)象的方法和屬性初始化值 ..... };
說明
1. tp_name
類型名, 這里是"type"
2. PyVarObject_HEAD_INIT(&PyType_Type, 0)
PyVarObject_HEAD_INIT, 這個(gè)方法在 Include/object.h中,
等價(jià)于
ob_refcnt = 1
*ob_type = &PyType_Type
ob_size = 0
即, PyType_Type的類型是其本身!
結(jié)構(gòu)
第一張圖, 箭頭表示實(shí)例化(google doc用不是很熟找不到對(duì)應(yīng)類型的箭頭)
第二張圖, 箭頭表示指向
使用
# 1. int 的 類型 是`type` >>> type(int) <type 'type'> # 2. type 的類型 還是`type`, 對(duì)應(yīng)上面說明第二點(diǎn) >>> type(type(int)) <type 'type'>
注意: 無論任何時(shí)候, ob_type指向的是 PyTypeObject的實(shí)例: PyType_Type/PyInt_Type...
3. 再然后, 定義具體的類型, 這里以PyInt_Type為例子
代碼位置 Objects/intobject.c
定義
PyTypeObject PyInt_Type = { PyVarObject_HEAD_INIT(&PyType_Type, 0) "int", sizeof(PyIntObject), 0, // int類型的相關(guān)方法和屬性值 .... (hashfunc)int_hash, /* tp_hash */ };
說明
1. "int"
PyInt_Type的類型名是int
2.PyVarObject_HEAD_INIT(&PyType_Type, 0)
PyInt_Type的
*ob_type = &PyType_Type
結(jié)構(gòu)
使用
>>> type(1) <type 'int'> >>> type(type(1)) <type 'type'>
4. 最后, 生成一個(gè)整數(shù)對(duì)象int
代碼位置 Include/intobject.h
定義
typedef struct { PyObject_HEAD long ob_ival; } PyIntObject;
結(jié)構(gòu)
- Python中的對(duì)象,方法,類,實(shí)例,函數(shù)用法分析
- 全面了解python中的類,對(duì)象,方法,屬性
- Python中的類與對(duì)象之描述符詳解
- Python入門篇之對(duì)象類型
- 在python的類中動(dòng)態(tài)添加屬性與生成對(duì)象
- Python中類的定義、繼承及使用對(duì)象實(shí)例詳解
- Python類的定義、繼承及類對(duì)象使用方法簡(jiǎn)明教程
- python基礎(chǔ)教程之對(duì)象和類的實(shí)際運(yùn)用
- Python面向?qū)ο缶幊讨械念惡蛯?duì)象學(xué)習(xí)教程
- 一篇文章搞懂Python的類與對(duì)象名稱空間
相關(guān)文章
python實(shí)現(xiàn)快速文件格式批量轉(zhuǎn)換的方法
這篇文章主要介紹了python實(shí)現(xiàn)快速文件格式批量轉(zhuǎn)換的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10python對(duì)輸出的奇數(shù)偶數(shù)排序?qū)嵗a
在本篇內(nèi)容里小編給大家整理的是一篇關(guān)于python對(duì)輸出的奇數(shù)偶數(shù)排序?qū)嵗a內(nèi)容,有興趣的朋友們可以參考下。2020-12-12Python常用Web框架Django、Flask與Tornado介紹
這篇文章介紹了Python常用Web框架Django、Flask與Tornado,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05python使用python-pptx刪除ppt某頁實(shí)例
今天小編就為大家分享一篇python使用python-pptx刪除ppt某頁實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-02-02pandas中apply和transform方法的性能比較及區(qū)別介紹
這篇文章主要介紹了pandas中apply和transform方法的性能比較,在文中給大家講解了apply() 與transform()的相同點(diǎn)與不同點(diǎn),需要的朋友可以參考下2018-10-10