广告
返回顶部
首页 > 资讯 > 后端开发 > Python >Python对象的生命周期源码学习
  • 327
分享到

Python对象的生命周期源码学习

2024-04-02 19:04:59 327人浏览 安东尼

Python 官方文档:入门教程 => 点击学习

摘要

目录思考:1 C api2 对象的创建2.1 两种创建对象的方式2.2 由类型对象创建实例对象3 对象的多态性4 对象的行为5 引用计数思考: 当我们输入这个语句的时候,python

思考:

当我们输入这个语句的时候,python内部是如何去创建这个对象的?

a = 1.0

对象使用完毕,销毁的时机又是怎么确定的呢?

下面,我们以一个基本类型float为例,来分析对象从创建到销毁这整个生命周期中的行为。

1 C API

Python是用C写的,对外提供了API,让用户可以从C环境中与其交互,并且Python内部也大量使用了这些API。C API分为两类:泛型API以及特型API。

泛型API:与类型无关,属于抽象对象层,这类API的参数是PyObject *,即可以处理任意类型的对象。以PyObject_Print为例:

// 打印浮点对象
PyObject *fo = PyFloat_FromDouble(3.14);
PyObject_Print(fo, stdout, 0);
// 打印整数对象
PyObject *lo = PyLong_FromLong(100);
PyObject_Print(lo, stdout, 0);

特型API:与类型相关,属于具体对象层,这类API只能作用于某种类型的对象

2 对象的创建

2.1 两种创建对象的方式

Python内部一般通过两种方法创建对象:

通过C API,多用于内建类型

以浮点类型为例,Python内部提供PyFloat_FromDouble,这是一个特型C API,在这个接口内部为PyFloatObject结构体变量分配内存,并初始化相关字段:

PyObject *
PyFloat_FromDouble(double fval)
{
    PyFloatObject *op = free_list;
    if (op != NULL) {
        free_list = (PyFloatObject *) Py_TYPE(op);
        numfree--;
    } else {
        op = (PyFloatObject*) PyObject_MALLOC(sizeof(PyFloatObject));
        if (!op)
            return PyErr_NoMemory();
    }
    
    (void)PyObject_INIT(op, &PyFloat_Type);
    op->ob_fval = fval;
    return (PyObject *) op;
}

通过类型对象,多用于自定义类型

对于自定义类型,Python就无法事先提供C API了,这种情况下就只能通过类型对象中包含的元数据(分配多少内存,如何初始化等等)来创建实例对象。

由类型对象创建实例对象是一个更通用的流程,对于内建类型,除了通过C API来创建对象意外,同样也可以通过类型对象来创建。以浮点类型为例,我们通过类型对象float,创建了一个实例对象f:

f: float = float('3.123')

2.2 由类型对象创建实例对象

思考:既然我们可以通过类型对象来创建实例对象,那么类型对象中应该存在相应的接口。

在PyType_Type中找到了tp_call字段:

PyTypeObject PyType_Type = {
    PyVarObject_HEAD_INIT(&PyType_Type, 0)
    "type",                                     
    sizeof(PyHeapTypeObject),                   
    sizeof(PyMemberDef),                        
    (destructor)type_dealloc,                   
    // ...
    (ternaryfunc)type_call,                     
    // ...
};

因此,float(‘3.123’)在C层面就等价于:

PyFloat_Type.ob_type.tp_call(&PyFloat_Type, args. kwargs)

这里大家可以思考下为什么是PyFloat_Type.ob_type——因为我们在float(‘3.14’)中是通过float这个类型对象去创建一个浮点对象,而对象的通用方法是由它对应的类型管理的,自然float的类型就是type,所以我们要找的就是type的tp_call字段。

type_call函数的C源码:(只列出部分)

static PyObject *
type_call(PyTypeObject *type, PyObject *args, PyObject *kwds)
{
    PyObject *obj;
    // ...
    obj = type->tp_new(type, args, kwds);
    obj = _Py_CheckFunctionResult((PyObject*)type, obj, NULL);
    if (obj == NULL)
        return NULL;
    // ...
    type = Py_TYPE(obj);
    if (type->tp_init != NULL) {
        int res = type->tp_init(obj, args, kwds);
        if (res < 0) {
            assert(PyErr_Occurred());
            Py_DECREF(obj);
            obj = NULL;
        }
        else {
            assert(!PyErr_Occurred());
        }
    }
    return obj;
}

其中有两个关键的步骤:(这两个步骤大家应该是很熟悉的)

  • 调用类型对象的tp_new函数指针,用于申请内存;
  • 如果类型对象的tp_init函数指针不为空,则会对对象进行初始化。

总结:(以float为例)

  • 调用float,Python最终执行的是其类型对象type的tp_call指针指向的type_call函数。
  • type_call函数调用float的tp_new函数为实例对象分配内存空间。
  • type_call函数必要时进一步调用tp_init函数对实例对象进行初始化。

图示如下:

3 对象的多态性

通过类型对象创建实例对象,最后会落实到调用type_call函数,其中保存具体对象时,使用的是PyObject *obj,并没有通过一个具体的对象(例如PyFloatObject)来保存。这样做的好处是:可以实现更抽象的上层逻辑,而不用关心对象的实际类型和实现细节。(记得当初从C语言的面向过程向Java中的面向对象过度的时候,应该就是从结构体)

以对象哈希值计算为例,有这样一个函数接口:

Py_hash_t
PyObject_Hash(PyObject *v)
{
    // ...
}

对于浮点数对象和整数对象:

PyObject *fo = PyFloatObject_FromDouble(3.14);
PyObject_Hash(fo);
PyObject *lo = PyLonGobject_FromLong(100);
PyObject_Hash(lo);

可以看到,对于浮点数对象和整数对象,我们计算对象的哈希值时,调用的都是PyObject_Hash()这个函数,但是对象类型不同,其行为是有区别的,哈希值计算也是如此。

那么在PyObject_Hash函数内部是如何区分的呢?

PyObject_Hash()函数具体逻辑:

Py_hash_t
PyObject_Hash(PyObject *v)
{
    PyTypeObject *tp = Py_TYPE(v);
    if (tp->tp_hash != NULL)
        return (*tp->tp_hash)(v);
    
    if (tp->tp_dict == NULL) {
        if (PyType_Ready(tp) < 0)
            return -1;
        if (tp->tp_hash != NULL)
            return (*tp->tp_hash)(v);
    }
    
    return PyObject_HashNotImplemented(v);
}

函数会首先通过Py_TYPE找到对象的类型,然后通过类型对象的tp_hash函数指针来调用对应的哈希计算函数。

即:PyObject_Hash()函数根据对象的类型,调用不同的函数版本,这就是多态。

4 对象的行为

除了tp_hash字段,PyTypeObject结构体还定义了很多函数指针,这些指针最终都会指向某个函数,或者为空。我们可以把这些函数指针看作是类型对象中定义的操作,这些操作决定了对应的实例对象在运行时的行为。

虽然不同的类型对象中保存了对应实例对象共有的行为,但是不同类型的对象也会存在一些共性。例如:整数对象和浮点数对象都支持加减乘除等擦欧总,元组对象和列表对象都支持下标操作。因此,我们以行为为分类标准,对对象进行分类:

Python以此为依据,为每个类别都定义了一个标准操作集:

  • PyNumberMethods结构体定义了数值型操作
  • PySequenceMethods结构体定义了序列型操作
  • PyMappingMethods结构体定义了关联型操作

如果类型对象提供了相关的操作集,则对应的实例对象就具备对应的行为:

typedef struct _typeobject {
    PyObject_VAR_HEAD
    const char *tp_name; 
    Py_ssize_t tp_basicsize, tp_itemsize; 
   // ...
    PyNumberMethods *tp_as_number;
    PySequenceMethods *tp_as_sequence;
    PyMappingMethods *tp_as_mapping;
    // ...
} PyTypeObject;

以float为例,类型对象PyFloat_Type的这三个字段是这样初始化的:

PyTypeObject PyFloat_Type = {
    PyVarObject_HEAD_INIT(&PyType_Type, 0)
    "float",
    sizeof(PyFloatObject),
    // ...
    &float_as_number,                           
    0,                                          
    0,                                          
    // ...
};

可以看到,只有tp_as_number非空,即float对象支持数值型操作,不支持序列型操作和关联型操作。

5 引用计数

在Python中,很多场景都涉及引用计数的调整:

  • 变量赋值
  • 函数参数传递
  • 属性操作
  • 容器操作

引用计数是Python生命周期中很关键的一个知识点,后续我会用一个单独的章节来介绍,这里咱们先按下不表,更多关于Python对象生命周期的资料请关注编程网其它相关文章!

--结束END--

本文标题: Python对象的生命周期源码学习

本文链接: https://www.lsjlt.com/news/118051.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Python对象的生命周期源码学习
    目录思考:1 C API2 对象的创建2.1 两种创建对象的方式2.2 由类型对象创建实例对象3 对象的多态性4 对象的行为5 引用计数思考: 当我们输入这个语句的时候,Python...
    99+
    2022-11-11
  • Python对象的生命周期源码分析
    本篇内容介绍了“Python对象的生命周期源码分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!思考:当我们输入这个语句的时候,Python...
    99+
    2023-06-30
  • 一起来学习Vue的生命周期
    目录生命周期生命周期的简单介绍beforeCreate与createdbeforeCreate()created()beforeMount与mountedbeforeMount()m...
    99+
    2022-11-13
  • SpringBoot源码之Bean的生命周期
    入口方法为SpringApplication#run() 1.SpringApplication#run() public ConfigurableApplicationCont...
    99+
    2023-05-15
    SpringBoot之Bean的生命周期 bean生命周期 SpringBean生命周期
  • Spring源码解析之Bean的生命周期
    一、Bean的实例化概述 前一篇分析了BeanDefinition的封装过程,最终将beanName与BeanDefinition以一对一映射关系放到beanDefinitionMa...
    99+
    2022-11-12
  • Java开发学习之Bean的生命周期详解
    目录一、什么是生命周期二、环境准备三、生命周期设置步骤1:添加初始化和销毁方法步骤2:配置生命周期步骤3:运行程序四、close关闭容器五、注册钩子关闭容器六、bean生命周期总结一...
    99+
    2022-11-13
  • Python对象的底层实现源码学习
    目录1. PyObject:对象的基石2. PyVarObject:变长对象的基础2.1 浮点对象2.2 列表对象3. PyTypeObject:类型的基石4. PyType_Typ...
    99+
    2022-11-11
  • SpringBoot源码之Bean的生命周期是什么
    本文小编为大家详细介绍“SpringBoot源码之Bean的生命周期是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“SpringBoot源码之Bean的生命周期是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知...
    99+
    2023-07-06
  • 用Flutter做APP学习心得:Flutter widget的生命周期
    第一次看文章的朋友可以关注我,会不定期发布大厂面试题、Android架构技术知识点及解析等内容,还有Android学习PDF+源码笔记+面试文档+进阶视频分享更多还可以看我的GitHub链接:https://github.com/Meng9...
    99+
    2023-06-04
  • C++临时性对象的生命周期详细解析
    有关临时对象的生命周期有三种情况:1)一般情况:临时性对象的被摧毁,应该是对完整表达式(full-expression)求值过程中的最后一个步骤。该完整表达式造成临时对象的产生。实例...
    99+
    2022-11-15
    C++ 临时性对象 生命周期
  • Python万物皆对象理解及源码学习
    目录万物皆对象1 类型对象和实例对象2 类型、对象体系2.1 元类型type2.2 自定义类型2.3 自定义类型子类2.4 type和object的关系3 可变对象与不可变对象4 变...
    99+
    2022-11-11
  • Java开发学习之Bean的作用域和生命周期详解
    目录一、Bean 的作用域二、Spring 的执行流程三、Bean 的生命周期一、Bean 的作用域 在之前学习Java基础的时候,有接触到作用域这样的概念。一个变量并不一定在任何区...
    99+
    2022-11-13
  • Python虚拟机栈帧对象及获取源码学习
    目录Python虚拟机1. 栈帧对象1.1 PyFrameObject1.2 栈帧对象链1.3 栈帧获取2. 字节码执行Python虚拟机 注:本篇是根据教程学习记录的笔记,部分内...
    99+
    2023-03-23
    Python虚拟机栈帧对象获取 Python虚拟机
  • Python中for循环可迭代对象迭代器及生成器源码学习
    目录问题:1. 迭代1.1 可迭代对象Iterable1.2 迭代器Iterator1.3 for循环1.3.1 iter()方法和next()方法1.3.2 iter()和__it...
    99+
    2022-11-11
  • react源码中的生命周期和事件系统实例解析
    目录引言jsx的编译结果React组件的生命周期组件挂载的时候的执行顺序组件更新的时候的执行顺序组件卸载的时候执行顺序组件在发生错误的时候执行顺序listenToAllSupport...
    99+
    2023-01-03
    react源码生命周期事件系统 react生命周期
  • WPF中的APP生命周期及全局异常捕获源码分析
    这篇文章主要讲解了“WPF中的APP生命周期及全局异常捕获源码分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“WPF中的APP生命周期及全局异常捕获源码分析”吧!APP生命周期wpf项目目...
    99+
    2023-07-05
  • Vue选项式API的生命周期选项和组合式API源码分析
    本文小编为大家详细介绍“Vue选项式API的生命周期选项和组合式API源码分析”,内容详细,步骤清晰,细节处理妥当,希望这篇“Vue选项式API的生命周期选项和组合式API源码分析”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来...
    99+
    2023-07-05
  • Java学习笔记:学会使用对象生成漂亮的二维码
    二维码是近年来非常流行的一种信息快速传递方式,它可以将一些文本信息、网址等内容编码成一张图像,方便用户在移动设备上进行扫描识别。在Java语言中,我们可以使用一些开源的库来生成漂亮的二维码图像。本文将介绍如何使用对象生成二维码,并提供一些...
    99+
    2023-08-13
    学习笔记 对象 二维码
  • 二维码是什么?Java学习笔记中的对象如何生成它?
    二维码是一种用于快速识别信息的编码方式,它由黑白相间的方块组成,可储存大量信息,而且读取速度快。在当今信息化的社会中,二维码已经成为了一种重要的信息传递方式。而在Java学习笔记中,我们可以通过一些对象来生成二维码。本篇文章将介绍二维码的基...
    99+
    2023-08-13
    学习笔记 对象 二维码
  • Python实现学生管理系统的完整代码(面向对象)
    前言 这个只是使用面向对象的方法写的 构思和学生管理系统(JSON模块)是一样的 file_manager.py """ Project: ClassStudent Creato...
    99+
    2022-11-12
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作