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

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

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

北京

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

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

看不清楚,换张图片

免费获取短信验证码

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虚拟机中元组的实现原理是什么”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

    免责声明:

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

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

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

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

    下载Word文档

    猜你喜欢

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

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

    Python虚拟机中列表的实现原理是什么

    这篇文章主要介绍“Python虚拟机中列表的实现原理是什么”,在日常操作中,相信很多人在Python虚拟机中列表的实现原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python虚拟机中列表的实现原理
    2023-07-05

    Python虚拟机中复数的实现原理是什么

    本篇内容主要讲解“Python虚拟机中复数的实现原理是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python虚拟机中复数的实现原理是什么”吧!复数数据结构在 cpython 当中对于复数
    2023-07-05

    Python虚拟机中字节的实现原理是什么

    这篇文章主要介绍“Python虚拟机中字节的实现原理是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Python虚拟机中字节的实现原理是什么”文章能帮助大家解决问题。数据结构typedef st
    2023-07-05

    Python虚拟机中整型的实现原理是什么

    这篇文章主要介绍“Python虚拟机中整型的实现原理是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Python虚拟机中整型的实现原理是什么”文章能帮助大家解决问题。数据结构在 cpython
    2023-07-05

    Python虚拟机中浮点数的实现原理是什么

    这篇文章主要介绍“Python虚拟机中浮点数的实现原理是什么”,在日常操作中,相信很多人在Python虚拟机中浮点数的实现原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python虚拟机中浮点数的实
    2023-07-05

    深入理解Python虚拟机中元组(tuple)的实现原理及源码

    在本篇文章当中主要给大家介绍 cpython 虚拟机当中针对列表的实现,在 Python 中,tuple 是一种非常常用的数据类型,在本篇文章当中将深入去分析这一点是如何实现的
    2023-03-11

    Python虚拟机集合set实现原理是什么

    本文小编为大家详细介绍“Python虚拟机集合set实现原理是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“Python虚拟机集合set实现原理是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。深入理解
    2023-07-05

    java中虚拟机jvm原理是什么

    这篇文章将为大家详细讲解有关java中虚拟机jvm原理是什么,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。曾几何时,我们还是初识Hello World的时候,我们哪曾知道,Java这门神奇的语言,在执行我
    2023-06-25

    java虚拟机的工作原理是什么

    Java虚拟机工作原理Java虚拟机(JVM)通过以下步骤执行Java程序:加载字节码验证字节码准备字节码编译字节码(JIT编译)执行字节码关键组件包括:Java堆、元空间、栈、垃圾回收器和类加载器。JVM将内存划分为运行时数据区域,包括程序计数器、虚拟机栈、本地方法栈、堆和方法区。类加载涉及加载、验证、准备、解析和初始化步骤。垃圾回收自动管理内存,释放未使用的对象。JVM优化包括JIT编译、垃圾回收和类加载优化。
    java虚拟机的工作原理是什么
    2024-04-12

    深入理解Python虚拟机中描述器的实现原理

    这篇文章主要给大家介绍一个我们在使用类的时候经常使用但是却很少在意的黑科技——描述器的实现原理,文中的示例代码讲解详细,需要的可以参考一下
    2023-05-19

    java虚拟主机运行的原理是什么

    Java虚拟主机(JVM)是一个虚拟的计算机,它运行在真实计算机上。JVM可以执行Java字节码文件,将其转换为可执行代码并在操作系统上运行。Java虚拟主机运行的原理如下:1. 通过Java编译器将Java源代码编译成Java字节码文件。
    2023-06-12

    java虚拟主机的工作原理是什么

    Java 虚拟主机(Java Virtual Hosting,JVH)是一种基于 Java 技术的虚拟主机服务,其工作原理如下:1、Java 虚拟主机的工作原理类似于传统的虚拟主机,但是其底层基于 Java 技术实现。2、Java 虚拟主机
    2023-03-20

    Java虚拟机中垃圾回收机制的原理是什么

    Java虚拟机中垃圾回收机制的原理是什么?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。在Java虚拟机中,对象和数组的内存都是在堆中分配的,垃圾收集器主要回收的
    2023-05-31

    基于域名的虚拟主机原理是什么

    基于域名的虚拟主机原理是通过将多个域名指向同一个物理服务器的不同目录或虚拟服务器,实现多个网站共享同一个服务器资源的技术。在传统的虚拟主机中,一台服务器通常只能绑定一个IP地址,因此只能承载一个域名的网站。而基于域名的虚拟主机通过HTTP协
    2023-09-07

    Python中hook的实现原理是什么

    在Python中,hook(钩子)是一种机制,允许开发者在特定事件(例如函数调用、异常发生等)发生时插入自定义的代码进行处理。实现原理主要基于Python的装饰器(Decorator)和元编程的概念。装饰器是Python中一种用来修饰函数或
    2023-09-26

    编程热搜

    • 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动态编译

    目录