返回顶部
首页 > 资讯 > 后端开发 > Python >Python虚拟机中元组的实现原理是什么
  • 714
分享到

Python虚拟机中元组的实现原理是什么

2023-07-05 11:07:04 714人浏览 泡泡鱼

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

摘要

这篇文章主要介绍了python虚拟机中元组的实现原理是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Python虚拟机中元组的实现原理是什么文章都会有所收获,下面我们一起来看看吧。元组的结构在这一小节当中主

这篇文章主要介绍了python虚拟机中元组的实现原理是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Python虚拟机中元组的实现原理是什么文章都会有所收获,下面我们一起来看看吧。

    元组的结构

    在这一小节当中主要介绍在 python 当中元组的数据结构

    typedef struct {    PyObject_VAR_HEAD    PyObject *ob_item[1];     } PyTupleObject; #define PyObject_VAR_HEAD      PyVarObject ob_base;typedef struct {    PyObject ob_base;    Py_ssize_t ob_size; } PyVarObject; typedef struct _object {    _PyObject_HEAD_EXTRA    Py_ssize_t ob_refcnt;    struct _typeobject *ob_type;} PyObject;

    从上面的数据结构来看和 list 的数据结构基本上差不多,最终的使用方法也差不多。将上面的结构体展开之后,PyTupleObject 的结构大致如下所示:

    Python虚拟机中元组的实现原理是什么

    现在来解释一下上面的各个字段的含义:

    • Py_ssize_t,一个整型数据类型。

    • ob_refcnt,表示对象的引用记数的个数,这个对于垃圾回收很有用处,后面我们分析虚拟机中垃圾回收部分在深入分析。

    • ob_type,表示这个对象的数据类型是什么,在 python 当中有时候需要对数据的数据类型进行判断比如 isinstance, type 这两个关键字就会使用到这个字段。

    • ob_size,这个字段表示这个元组当中有多少个元素。

    • ob_item,这是一个指针,指向真正保存 python 对象数据的地址,大致的内存他们之间大致的内存布局如下所示:

    Python虚拟机中元组的实现原理是什么

    需要注意的是元组的数组大小是不能够进行更改的,这一点和 list 不一样,我们可以注意到在 list 的数据结构当中还有一个 allocated 字段,但是在元组当中是没有的,这主要是因为元组的数组大小是固定的,而列表的数组大小是可以更改的。

    元组操作函数源码剖析

    创建元组

    首先我们需要了解一下在 cpython 内部关于元组内存分配的问题,首先和 list 一样,在 cpython 当中对于分配的好的元组进行释放的时候,并不会直接进行释放,而是会先保存下来,当下次又有元组申请内存的时候,直接将这块内存进行返回即可。

    在 cpython 内部会进行缓存的元组大小为 20,如果元组的长度为 0 - 19 那么在申请分配内存之后释放并不会直接释放,而是将其先保存下来,下次有需求的时候直接分配,而不需要申请。在 cpython 内部,相关的定义如下所示:

    static PyTupleObject *free_list[PyTuple_MAXSAVESIZE];static int numfree[PyTuple_MAXSAVESIZE];
    • free_list,保存指针——指向被释放的元组。

    • numfree,对应的下标表示元组当中元素的个数,numfree[i] 表示有 i 个元素的元组的个数。

    下面是新建 tuple 对象的源程序:

    PyObject *PyTuple_New(Py_ssize_t size){    PyTupleObject *op;    Py_ssize_t i;    if (size < 0) {        PyErr_BadInternalCall();        return NULL;    }#if PyTuple_MAXSAVESIZE > 0    // 如果申请一个空的元组对象 当前的 free_list 当中是否存在空元组对象 如果存在则直接返回    if (size == 0 && free_list[0]) k        op = free_list[0];        Py_INCREF(op);        return (PyObject *) op;    }    // 如果元组的对象元素个数小于 20 而且对应的 free_list 当中还有余下的元组对象 则不需要进行内存申请直接返回    if (size < PyTuple_MAXSAVESIZE && (op = free_list[size]) != NULL) {        free_list[size] = (PyTupleObject *) op->ob_item[0];        numfree[size]--;                _Py_NewReference((PyObject *)op); // _Py_NewReference 这个宏是将对象 op 的引用计数设置成 1    }    else#endif    {                // 如果元组的元素个数大或者等于 20 或者 当前 free_list 当中没有没有剩余的对象则需要进行内存申请        if ((size_t)size > ((size_t)PY_SSIZE_T_MAX - sizeof(PyTupleObject) -                    sizeof(PyObject *)) / sizeof(PyObject *)) {          // 如果元组长度大于某个值直接报内存错误            return PyErr_NoMemory();        }        // 申请元组大小的内存空间        op = PyObject_GC_NewVar(PyTupleObject, &PyTuple_Type, size);        if (op == NULL)            return NULL;    }// 初始化内存空间    for (i=0; i < size; i++)        op->ob_item[i] = NULL;#if PyTuple_MAXSAVESIZE > 0    // 因为 size == 0 的元组不会进行修改操作 因此可以直接将这个申请到的对象放到 free_list 当中以备后续使用    if (size == 0) {        free_list[0] = op;        ++numfree[0];        Py_INCREF(op);              }#endif    _PyObject_GC_TRACK(op); // _PyObject_GC_TRACK 这个宏是将对象 op 将入到垃圾回收队列当中    return (PyObject *) op;}

    新建元组对象的流程如下所示:

    • 查看 free_list 当中是否已经存在空闲的元组,如果有则直接进行返回。

    • 如果没有,则进行内存分配,然后将申请的内存空间进行初始化操作。

    • 如果 size == 0,则可以将新分配的元组对象放到 free_list 当中。

    查看元组的长度

    这个功能比较简单,直接只用 cpython 当中的宏 Py_SIZE 即可。他的宏定义为 #define Py_SIZE(ob) (((PyVarObject*)(ob))->ob_size)。

    static Py_ssize_ttuplelength(PyTupleObject *a){    return Py_SIZE(a);}

    元组当中是否包含数据

    这个其实和 list 一样,就是遍历元组当中的数据,然后进行比较即可。

    static inttuplecontains(PyTupleObject *a, PyObject *el){    Py_ssize_t i;    int cmp;     for (i = 0, cmp = 0 ; cmp == 0 && i < Py_SIZE(a); ++i)        cmp = PyObject_RichCompareBool(el, PyTuple_GET_ITEM(a, i),                                           Py_EQ);    return cmp;}

    获取和设置元组中的数据

    这两个方法也比较简单,首先检查数据类型是不是元组类型,然后判断是否越界,之后就返回数据,或者设置对应的数据。

    这里在设置数据数据的时候需要注意一点的是,当设置新的数据的时候,原来的 python 对象引用计数需要减去一,同理如果设置没有成功的话传入的新的数据的引用计数也需要减去一。

    PyObject *PyTuple_GetItem(PyObject *op, Py_ssize_t i){    if (!PyTuple_Check(op)) {        PyErr_BadInternalCall();        return NULL;    }    if (i < 0 || i >= Py_SIZE(op)) {        PyErr_SetString(PyExc_IndexError, "tuple index out of range");        return NULL;    }    return ((PyTupleObject *)op) -> ob_item[i];} intPyTuple_SetItem(PyObject *op, Py_ssize_t i, PyObject *newitem){    PyObject *olditem;    PyObject **p;    if (!PyTuple_Check(op) || op->ob_refcnt != 1) {        Py_XDECREF(newitem);        PyErr_BadInternalCall();        return -1;    }    if (i < 0 || i >= Py_SIZE(op)) {        Py_XDECREF(newitem);        PyErr_SetString(PyExc_IndexError,                        "tuple assignment index out of range");        return -1;    }    p = ((PyTupleObject *)op) -> ob_item + i;    olditem = *p;    *p = newitem;    Py_XDECREF(olditem);    return 0;}

    释放元组内存空间

    当我们在进行垃圾回收的时候,判定一个对象的引用计数等于 0 的时候就需要释放这块内存空间(相当于析构函数),下面就是释放 tuple 内存空间的函数。

    static voidtupledealloc(PyTupleObject *op){    Py_ssize_t i;    Py_ssize_t len =  Py_SIZE(op);    PyObject_GC_UnTrack(op); // PyObject_GC_UnTrack 将对象从垃圾回收队列当中移除    Py_TRASHCAN_SAFE_BEGIN(op)     if (len > 0) {        i = len;        while (--i >= 0)            // 将这个元组指向的对象的引用计数减去一            Py_XDECREF(op->ob_item[i]);#if PyTuple_MAXSAVESIZE > 0        // 如果这个元组对象满足加入 free_list  的条件,则将这个元组对象加入到 free_list 当中        if (len < PyTuple_MAXSAVESIZE &&            numfree[len] < PyTuple_MAXFREELIST &&            Py_TYPE(op) == &PyTuple_Type)        {            op->ob_item[0] = (PyObject *) free_list[len];            numfree[len]++;            free_list[len] = op;            Goto done;         }#endif    }    Py_TYPE(op)->tp_free((PyObject *)op);done:    Py_TRASHCAN_SAFE_END(op)}

    将元组的内存空间回收的时候,主要有以下几个步骤:

    • 将元组对象从垃圾回收链表当中移除。

    • 将元组指向的所有对象的引用计数减一。

    • 判断元组是否满足保存到 free_list 当中的条件,如果满足就将他加入到 free_list 当中去,否则回收这块内存。加入到 free_list 当中整个元组当中 ob_item 指向变化如下所示:

    Python虚拟机中元组的实现原理是什么

    如果不能够将释放的元组对象加入到 free_list 当中,否则将内存释放回收。

    关于“Python虚拟机中元组的实现原理是什么”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“Python虚拟机中元组的实现原理是什么”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网Python频道。

    --结束END--

    本文标题: Python虚拟机中元组的实现原理是什么

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

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

    猜你喜欢
    • Python虚拟机中元组的实现原理是什么
      这篇文章主要介绍了Python虚拟机中元组的实现原理是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Python虚拟机中元组的实现原理是什么文章都会有所收获,下面我们一起来看看吧。元组的结构在这一小节当中主...
      99+
      2023-07-05
    • Python虚拟机中字典的实现原理是什么
      字典数据结构分析 typedef struct { PyObject_HEAD Py_ssize_t ma_used; PyDictKeysObject *ma_keys; PyObject **ma_val...
      99+
      2023-05-19
      Python
    • Python虚拟机中列表的实现原理是什么
      这篇文章主要介绍“Python虚拟机中列表的实现原理是什么”,在日常操作中,相信很多人在Python虚拟机中列表的实现原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python虚拟机中列表的实现原理...
      99+
      2023-07-05
    • Python虚拟机中字节的实现原理是什么
      这篇文章主要介绍“Python虚拟机中字节的实现原理是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Python虚拟机中字节的实现原理是什么”文章能帮助大家解决问题。数据结构typedef&nb...
      99+
      2023-07-05
    • Python虚拟机中复数的实现原理是什么
      本篇内容主要讲解“Python虚拟机中复数的实现原理是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python虚拟机中复数的实现原理是什么”吧!复数数据结构在 cpython 当中对于复数...
      99+
      2023-07-05
    • Python虚拟机中整型的实现原理是什么
      这篇文章主要介绍“Python虚拟机中整型的实现原理是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Python虚拟机中整型的实现原理是什么”文章能帮助大家解决问题。数据结构在 cpython ...
      99+
      2023-07-05
    • Python虚拟机中调试器的实现原理是什么
      调试器是一个编程语言非常重要的部分,调试器是一种用于诊断和修复代码错误(或称为 bug)的工具,它允许开发者在程序执行时逐步查看和分析代码的状态和行为,它可以帮助开发者诊断和修复代码错误,理解程序的行为,优化性能。调试器在任何编程语言中都是...
      99+
      2023-05-21
      Python
    • Python虚拟机中浮点数的实现原理是什么
      这篇文章主要介绍“Python虚拟机中浮点数的实现原理是什么”,在日常操作中,相信很多人在Python虚拟机中浮点数的实现原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python虚拟机中浮点数的实...
      99+
      2023-07-05
    • 深入理解Python虚拟机中元组(tuple)的实现原理及源码
      目录元组的结构元组操作函数源码剖析创建元组查看元组的长度元组当中是否包含数据获取和设置元组中的数据释放元组内存空间总结元组的结构 在这一小节当中主要介绍在 python 当中元组的数...
      99+
      2023-03-11
      Python 虚拟机 元组 Python虚拟机 Python 元组
    • Python虚拟机集合set实现原理是什么
      本文小编为大家详细介绍“Python虚拟机集合set实现原理是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“Python虚拟机集合set实现原理是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。深入理解 ...
      99+
      2023-07-05
    • java中虚拟机jvm原理是什么
      这篇文章将为大家详细讲解有关java中虚拟机jvm原理是什么,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。曾几何时,我们还是初识Hello World的时候,我们哪曾知道,Java这门神奇的语言,在执行我...
      99+
      2023-06-25
    • android虚拟机原理是什么
      Android虚拟机的原理是将Android操作系统安装在主机操作系统上,通过虚拟化技术实现在主机上运行Android应用程序。具体...
      99+
      2023-10-12
      android
    • 深入理解Python虚拟机中描述器的实现原理
      目录描述器的基本用法描述器的实现原理描述器的应用场景实现属性访问控制实现数据验证和转换实现延迟加载和缓存实现 ORM 映射总结在本篇文章当中主要给大家介绍一个我们在使用类的时候经常使...
      99+
      2023-05-19
      Python虚拟机描述器实现原理 Python 描述器 Python虚拟机
    • java虚拟主机运行的原理是什么
      Java虚拟主机(JVM)是一个虚拟的计算机,它运行在真实计算机上。JVM可以执行Java字节码文件,将其转换为可执行代码并在操作系...
      99+
      2023-06-12
      java虚拟主机 虚拟主机
    • java虚拟主机的工作原理是什么
      Java 虚拟主机(Java Virtual Hosting,JVH)是一种基于 Java 技术的虚拟主机服务,其工作原理如下:1、...
      99+
      2023-03-20
      java虚拟主机 虚拟主机
    • Java虚拟机中垃圾回收机制的原理是什么
      Java虚拟机中垃圾回收机制的原理是什么?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。在Java虚拟机中,对象和数组的内存都是在堆中分配的,垃圾收集器主要回收的...
      99+
      2023-05-31
      虚拟机 垃圾回收机制 java
    • docker虚拟化的原理是什么
      Docker 虚拟化的原理主要是基于 Linux 内核的 cgroups(控制组)和 namespaces(命名空间)技术实现的。 ...
      99+
      2024-04-09
      docker
    • 基于域名的虚拟主机原理是什么
      基于域名的虚拟主机原理是通过将多个域名指向同一个物理服务器的不同目录或虚拟服务器,实现多个网站共享同一个服务器资源的技术。在传统的虚...
      99+
      2023-09-07
      虚拟主机
    • Python中hook的实现原理是什么
      在Python中,hook(钩子)是一种机制,允许开发者在特定事件(例如函数调用、异常发生等)发生时插入自定义的代码进行处理。实现原...
      99+
      2023-09-26
      Python
    • Python中Dict实现的原理是什么
      1.无序Dict的实现Dict能够快速查找key,这归功于它采用的空间换时间策略和哈希表实现。的在读取和写入Key时, 都会对Key进行哈希计算(所以要求Key都是不可变类型,如果是可变类型,就无法计算出他的哈希值了), 然后根据计算的值,...
      99+
      2023-05-19
      Python dict
    软考高级职称资格查询
    编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
    • 官方手机版

    • 微信公众号

    • 商务合作