我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Python对象的生命周期源码分析

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Python对象的生命周期源码分析

本篇内容介绍了“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函数对实例对象进行初始化。

    图示如下:

    Python对象的生命周期源码分析

    3 对象的多态性

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

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

    Py_hash_tPyObject_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_tPyObject_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对象的生命周期源码分析

    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对象的生命周期源码分析”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

    免责声明:

    ① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

    ② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

    Python对象的生命周期源码分析

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

    下载Word文档

    猜你喜欢

    Python对象的生命周期源码分析

    本篇内容介绍了“Python对象的生命周期源码分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!思考:当我们输入这个语句的时候,Python
    2023-06-30

    rust生命周期源码分析

    本文小编为大家详细介绍“rust生命周期源码分析”,内容详细,步骤清晰,细节处理妥当,希望这篇“rust生命周期源码分析”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。rust生命周期生命周期是rust中用来规定引
    2023-07-05

    Spring Bean生命周期源码分析

    这篇“Spring Bean生命周期源码分析”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Spring Bean生命周期源码
    2023-07-05

    react中context传值和生命周期源码分析

    本篇内容主要讲解“react中context传值和生命周期源码分析”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“react中context传值和生命周期源码分析”吧!假设:项目中存在复杂组件树:
    2023-07-05

    Vue八大生命周期钩子函数源码分析

    本篇内容主要讲解“Vue八大生命周期钩子函数源码分析”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Vue八大生命周期钩子函数源码分析”吧!一.速识概念:我们把一个对象从生成(new)到被销毁(d
    2023-07-05

    SpringBoot源码之Bean的生命周期

    spring的bean的生命周期主要是创建bean的过程,一个bean的生命周期主要是4个步骤,实例化,属性注入,初始化,销毁,本文详细介绍了bean的生命周期,感兴趣的小伙伴可以参考阅读
    2023-05-15

    Java之Spring Bean作用域和生命周期源码分析

    这篇文章主要讲解了“Java之Spring Bean作用域和生命周期源码分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java之Spring Bean作用域和生命周期源码分析”吧!Bea
    2023-07-05

    微信小程序开发中源码分析生命周期

    这篇文章主要介绍了微信小程序开发中源码分析生命周期的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇微信小程序开发中源码分析生命周期文章都会有所收获,下面我们一起来看看吧。生命周期的概念在讲微信小程序生命周期之前,
    2023-07-05

    C++临时性对象的生命周期详细解析

    临时性对象的被摧毁,应该是对完整表达式(full-expression)求值过程中的最后一个步骤。该完整表达式造成临时对象的产生
    2022-11-15

    Laravel的生命周期实例分析

    本篇内容主要讲解“Laravel的生命周期实例分析”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Laravel的生命周期实例分析”吧!Laravel的生命周期 A世间万物皆有生命周期,当我们使用
    2023-06-30

    vue生命周期的示例分析

    这篇文章主要介绍了vue生命周期的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。vue生命周期图解感谢你能够认真阅读完这篇文章,希望小编分享的“vue生命周期的示例分
    2023-06-14

    React的生命周期实例分析

    这篇文章主要讲解了“React的生命周期实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“React的生命周期实例分析”吧!一、React生命周期React 生命周期分为三种状态1. 初
    2023-07-02

    WPF中的APP生命周期及全局异常捕获源码分析

    这篇文章主要讲解了“WPF中的APP生命周期及全局异常捕获源码分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“WPF中的APP生命周期及全局异常捕获源码分析”吧!APP生命周期wpf项目目
    2023-07-05

    React中生命周期的示例分析

    这篇文章将为大家详细讲解有关React中生命周期的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。React的生命周期两张图带你理解 React的生命周期React的生命周期(旧)class Lif
    2023-06-20

    编程热搜

    • Python 学习之路 - Python
      一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
      Python 学习之路 - Python
    • chatgpt的中文全称是什么
      chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
      chatgpt的中文全称是什么
    • C/C++中extern函数使用详解
    • C/C++可变参数的使用
      可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
      C/C++可变参数的使用
    • css样式文件该放在哪里
    • php中数组下标必须是连续的吗
    • Python 3 教程
      Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
      Python 3 教程
    • Python pip包管理
      一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
      Python pip包管理
    • ubuntu如何重新编译内核
    • 改善Java代码之慎用java动态编译

    目录