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

Python垃圾回收机制的原理

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

北京

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

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

看不清楚,换张图片

免费获取短信验证码

Python垃圾回收机制的原理

本篇内容介绍了“Python垃圾回收机制的原理”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

  引用计数器为主

  标记清除和分代回收为辅

  + 缓存机制

  1. 引用计数器

  1.1 环状双向链表 refchain

  在python程序中创建的任何对象都会放在refchain链表中。

  #define PyObject_HEAD PyObject ob_base;

  #define PyObject_VAR_HEAD PyVarObject ob_base;

  // 宏定义,包含 上一个、下一个,用于构造双向链表用。(放到refchain链表中时要用到)

  #define _PyObject_HEAD_EXTRA \

  struct _object *_ob_next; \

  struct _object *_ob_prev;

  name = "阿玮"

  age = 18

  hobby = ["健身", "美女"]

  内部会创建一些数据 [ 上一个对象、下一个对象、类型、引用个数 ]

  name = "阿玮"

  new = name # 引用个数变成2

  内部会创建一些数据 [ 上一个对象、下一个对象、类型、引用个数、val=18 ]

  age = 18

  内部会创建一些数据 [ 上一个对象、下一个对象、类型、引用个数、items=元素、元素个数 ]

  hobby = ["健身", "美女"]

  #define PyObject_HEAD PyObject ob_base;

  #define PyObject_VAR_HEAD PyVarObject ob_base;

  // 宏定义,包含 上一个、下一个,用于构造双向链表用。(放到refchain链表中时要用到)

  #define _PyObject_HEAD_EXTRA \

  struct _object *_ob_next; \

  struct _object *_ob_prev;

  typedef struct _object {

  _PyObject_HEAD_EXTRA; // 用于构造双向链表

  Py_ssize_t ob_refcnt; // 引用计数器

  struct _typeobject *ob_type; // 数据类型

  } PyObject;

  typedef struct {

  PyObject ob_base; // PyObject对象

  Py_ssize_t ob_size; // Number of items in variable part,即:元素个数

  } PyVarObject;

  在C源码中如何体现每个对象中都有的相同的值:PyObject结构体(4个值)。

  有多个元素组成的对象:PyObject结构体(4个值)+ ob_size = PyVarObject。

  1.2 类型封装结构体

  float类型

  typedef struct {

  PyObject_HEAD;

  double ob_fval;

  };

  data = 3.14;

  内部会创建:

  _ob_next = refchain中的下一个对象

  _ob_prev = refchain中的上一个对象

  ob_refcnt = 1

  ob_type = float

  ob_fval = 3.14

  int类型

  struct _longobect {

  PyObject_VAR_HEAD;

  digit ob_dit[1];

  };

  

  typedef struct _longobject PyLongObject;

  list类型

  typedef struct {

  PyObject_VAR_HEAD;

  PyObject ** ob_item;

  Py_ssize_t allocated;

  } PyListObject;

  tuple类型

  typedef struct {

  PyObject_VAR_HEAD;

  PyObject *ob_item[1];

  } PyTupleObject;

  dict类型

  typedef struct {

  PyObject_HEAD;

  Py_ssize_t ma_used;

  PyDictKeyObject *ma_keys;

  PyObject **ma_values;

  } PyDictObject;

  1.3 引用计数器

  v1 = 3.14

  v2 = 999

  v3 = (1,2,3)

  当python程序运行时,会根据数据类型的不同找到其结构体,根据结构体中的字段来进行创建相关的数据,然后将对象添加到refchain双向链表中。

  在C源码中有两个关键的结构体:PyObject、PyVarObject。

  每个对象中有 ob_refcnt 就是引用计数器,值默认为1,当有其他变量引用这个对象时,引用计数器就会发生变化。

  引用

  a = 99999

  b = a

  # 此时 99999 这个对象引用计数器的值为2

  '''

  下面情况会导致引用计数器+1:

  1.对象被创建,如 a = 2

  2.对象被引用,如 b = a

  3.对象被作为参数,传入到一个函数中

  4.对象作为一个元素,存储在容器中

  可以通过sys包中的getrefcount()来获取一个名称所引用的对象当前的引用计数器的值(注意这里getrefcount()本身会使得引用计数器+1)

  '''

  删除引用

  a = 99999

  b = a

  # b变量删除,b对应对象的引用计数器-1

  def b

  # a变量删除,a对应对象的引用计数器-1

  '''

  下面情况会导致引用计数器-1:

  1.变量被显示销毁 del

  2.变量被赋予新的对象

  3.一个对象离开它的作用域

  4.对象所在的容器被销毁或从容器中删除对象

  '''

  # 当一个对象的引用计数器为0时,意味着没有人再使用这个对象了,这个对象就是垃圾,垃圾回收。

  # 回收:1.对象从rechain链表移出。2.将对象销毁,内存归还。

  1.4 循环引用问题

  由于 v1 指向的对象引用了 v2,v2 指向的对象也引用了 v1,当将 v1、v2 两个变量删除时,虽然引用计数器会减1,但是两个对象间还存在循环引用,而此时已经没有变量能去指向它们,这两个对象就会在内存中常驻无法处理。

  2. 标记清除

  目的:为了解决引用计数器循环引用的问题。

  实现:在python的底层再维护一个链表,链表中专门放哪些可能存在循环应用的对象(容器类对象:list、tuple、dict、set)。

  在Python内部某种情况下触发,会去扫描可能存在循环引用的链表中的每个元素,检查是否有循环引用,如果有则让双方的引用计数器-1;如果是0则垃圾回收。

  2.1 标记阶段

  遍历所有对象,如果是可达的(reachable),也就是还有对象引用它,那么就将该对象标记为可达

  该阶段从某个对象开始扫描(而不是从变量),如果变量A引用了变量B,则将变量B的引用计数器-1(指的是gc_ref),然后扫描变量B郑州人流手术医院 http://wap.zyfuke.com/

  如图所示,link1、link2、link3形成了一个引用环,link4自引用。从link1开始扫描,link1引用了link2,则link2的gc_ref-1,接着扫描link2…

  像这也将链表中所有对象考察一遍后,两个链表中的对象ref_count和gc_ref图如所示,这一步操作就相当于解除了循环引用对引用计数器的影响

  如果gc_ref为0,则将对象标记为 GC_TENTATIVELY_UNREACHABLE,并且被移至”Unreachable“链表中,如下图link3、link4(我觉得link2应该也是)

  如果gc_ref不为0,那么这个对象会被标记为可达的GC_REACHABLE,同时当gc发现有一个节点是可达的,那么它会递归式的从该节点触发将所有可达的节点标记为GC_REACHABLE,这样把link2、link3救回来

  2.2 清除阶段

  将被标记成 GC_UNREACHABLE 的对象销毁,内存归还(也就是Unreachable链表中的对象)

  2.3 标记清除的问题

  在标记清除算法开始后,会暂停整个应用程序,等待标记清除结束后才会恢复应用的运行,且对循环引用的扫描代价大,每次扫描耗时可能很久

  3. 分代回收

  将可能存在循环引用的对象维护成3个链表:

  0代:0代中对象个数达到700个扫描一次

  1代:0代扫描10次,则1代扫描一次

  2代:1代扫描10次,则2代扫描一次

  4. 小结

  在python中维护了一个refchain的双向环状链表,这个链表中存储程序创建的所有对象,每种类型的对象都有一个ob_refcnt引用计数器的值,当引用计数器变为0时会进行垃圾回收(对象销毁、refchain中移出)。

  但是,在python中对于那些可以有多个元素组成的对象可能会存在循环引用的问题,为了解决这个问题,python又引入了标记清除和分代回收,在其内部维护了4个链表,分别为:

  refchain

  2代

  1代

  0代

  在源码内部,当达到各自的阈值时,就会触发扫描链表进行标记清除的动作(有循环引用则各自-1)。

  But,源码内部在上述流程中提出了优化机制。

“Python垃圾回收机制的原理”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

免责声明:

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

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

Python垃圾回收机制的原理

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

下载Word文档

猜你喜欢

Python垃圾回收机制的原理

本篇内容介绍了“Python垃圾回收机制的原理”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!  引用计数器为主  标记清除和分代回收为辅  
2023-06-01

Python垃圾回收机制的原理是什么

Python的垃圾回收机制是自动进行的,它基于引用计数的原理以及循环垃圾收集。引用计数:Python中的每个对象都有一个引用计数值,表示指向该对象的引用个数。当创建一个对象时,引用计数为1。当对象被引用时,引用计数增加;当对象不再被引用时,
2023-10-20

理解Python垃圾回收机制

一.垃圾回收机制 Python中的垃圾回收是以引用计数为主,分代收集为辅。引用计数的缺陷是循环引用的问题。 在Python中,如果一个对象的引用数为0,Python虚拟机就会回收这个对象的内存。#encoding=utf-8 __autho
2022-06-04

python垃圾回收机制!

python的三种垃圾回收机制:1.python采用的是引用计数机制为主;2.标记-清除;为辅的策略3.分代收集(隔代回收、分代回收)为辅的策略现在的高级语言如java,c#等,都采用了垃圾收集机制,而不再是c,c++里用户自己管理维护内存
2023-01-31

Java 垃圾回收机制的原理究竟是什么?(java垃圾回收机制的原理是什么)

在Java编程中,垃圾回收机制是一个非常重要的概念。它负责自动管理内存,回收不再被引用的对象,以避免内存泄漏和提高程序的性能。本文将详细介绍Java垃圾回收机制的原理。一、什么是垃圾回收垃圾回收是一种自动内存管理技术,它
Java 垃圾回收机制的原理究竟是什么?(java垃圾回收机制的原理是什么)
Java2024-12-22

垃圾回收机制

我们定义变量会申请内存空间来存放变量的值,而内存的容量是有限的,当一个变量值没有用了(称为垃圾),就应该将其占用的内存给回收掉。变量名是访问到变量的唯一方式,所以当一个变量值没有任何关联的变量名时,我们就无法访问到该变量了,该变量就是一个垃
2023-01-30

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

本篇内容主要讲解“Java垃圾回收机制的原理是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java垃圾回收机制的原理是什么”吧!Java 垃圾回收机制1. 垃圾回收主要关注 Java 堆图
2023-06-15

js垃圾回收机制的原理是什么

这篇“js垃圾回收机制的原理是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“js垃圾回收机制的原理是什么”文章吧。前言大
2023-06-27

jvm垃圾回收机制的原理是什么

JVM(Java虚拟机)的垃圾回收机制是自动管理内存的一种机制,它通过识别和释放不再被程序使用的对象,以防止内存泄漏和内存溢出等问题。JVM的垃圾回收机制基于以下原理:1. 引用计数:JVM使用引用计数来跟踪对象的引用数。每当一个对象被引用
2023-09-25

浅谈Python的垃圾回收机制

一.垃圾回收机制 Python中的垃圾回收是以引用计数为主,分代收集为辅。引用计数的缺陷是循环引用的问题。 在Python中,如果一个对象的引用数为0,Python虚拟机就会回收这个对象的内存。#encoding=utf-8 __autho
2022-06-04

PHP的垃圾回收机制的原理是什么

这篇文章将为大家详细讲解有关PHP的垃圾回收机制的原理是什么,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。PHP的基本GC概念PHP语言同其他语言一样,具有垃圾回收机制。那么今天我们要为大家
2023-06-17

PHP的垃圾回收机制

PHP垃圾回收机制PHP采用引用计数机制管理内存,跟踪变量指向对象的次数。当引用计数降至0时,对象将被释放。循环引用会阻止对象释放,导致内存泄漏。PHP还提供弱引用、对象销毁器和垃圾回收循环等机制来帮助清理垃圾。虽然垃圾回收机制通常有效,但在处理大数据集时或出现循环引用时可能会影响性能。优化垃圾回收可以通过避免循环引用、使用弱引用和显式garbagecollection等措施来实现。
PHP的垃圾回收机制
2024-04-25

PHP的垃圾回收机制

一、原理 php5和php7的垃圾回收机制都是利用引用计数。 二、php5和php7不同点 1、PHP5标量数据类型会计数,PHP7标量数据类型不再计数,不需要单独分配内存。 2、PHP7的zval 需要的内存不再是单独从堆上分配,不再自己存储引用计数。 3、
PHP的垃圾回收机制
2016-04-04

day09(垃圾回收机制)

1,复习文件处理1.操作文件的三步骤 -- 打开文件:硬盘的空间被操作系统持有 | 文件对象被应用程序持续 -- 操作文件:读写操作 -- 释放文件:释放操作系统对硬盘空间的持有 2.基础的读写with open('
2023-01-31

一文读懂Java垃圾回收机制原理

Java 内存运行时区域中的程序计数器、虚拟机栈、本地方法栈随线程而生灭;栈中的栈帧随着方法的进入和退出而有条不紊地执行着出栈和入栈操作。

编程热搜

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

目录