详细分析Python垃圾回收机制

作者:云崖先生 时间:2021-04-18 02:43:43 

引入

为什么要有垃圾回收机制

Python中的垃圾回收机制简称(GC),我们在程序的运行中会产生大量的变量用于保存数据,而有时候有些变量已经没有用了就需要被清理释放掉该变量所占据的内存空间。在一些较为低级的语言中(比如:C语言,汇编语言)对于内存空间的释放是需要编程人员来手动进行的,这种与底层硬件直接打交道的操作是十分的危险与繁琐的,而基于C语言开发而来的Python为了解决掉这种顾虑则自带了一种垃圾回收机制,从而让开发人员不必过分担心内存的使用情况而可以全身心的投入到开发中去。


>>> name = "yunya" #yunya 准备改名
>>> name = "yunyaya" #原本yunya这个名字不使用了,现在必须清理掉它否则将会占据内存空间,所幸Python的垃圾回收机制会帮我清理掉 "yunya"
>>

堆区和栈区的概念

如果你看我之前写的那篇文章关于Python变量的底层原理的话那么想必对堆区和栈区内存有了一定的了解。如果没有看过那么也没有关系,链接如下:

Python变量与基本数据类型

底层工作原理

引用计数

引用计数说白了就是来对堆区的变量值绑定的栈区变量名来计数。如图:

详细分析Python垃圾回收机制

当使用del或者对变量名重新赋值后,该变量值的引用计数就会 -1 。当引用计数为 0 时候下次 Python内存回收机制 进行内存扫描时便会将该变量值当做垃圾进行回收。

详细分析Python垃圾回收机制

那么这里就是Python内存回收机制中最基本的也最常用的引用计数介绍。

循环引用-内存泄漏

引用计数虽然作为Python内存回收机制中最经常使用的一种机制,但是它本身也是具有一定的缺点。我们来看下面这段代码:


>>> l1 = [1,2,3]
>>> l2 = [1,2,3,l1]
>>> l1.append(l2)  #append()方法用于向列表中添加一个元素值
>>> l1
[1, 2, 3, [1, 2, 3, [...]]]
>>> l2
[1, 2, 3, [1, 2, 3, [...]]]
>>>

现在l1和l2全部作为互相引用了。那么对于这种引用方式叫做循环引用(也被称为交叉引用),循环引用会带来一个问题:

  1. l1 变量值 的引用计数 目前为 2

  2. l2 变量值 的引用计数 目前为 2

  3. 当使用 del l1 与 del l2 后呢?

  4. 它们的引用变量都减1,但是引用方式的变量名都互相删除了,按理说这些变量值都成了垃圾变量。单根据引用计数是无法清理这些垃圾变量的。

 详细分析Python垃圾回收机制


>>> del l1
>>> del l2
>>> #现在怎么访问 li1 或者 li2 呢?访问不到,但是他们的变量值依然存在于内存,引用计数从2变为1

标记-清除

标记清除的意思在于当应用程序可用内存空间即将被耗尽时便开始扫描栈区,并且会顺着栈区变量名对堆区中的变量值做一个标记,如果堆区中存在没有与栈区变量名做对应关系的数据则会被认为是垃圾数据从而被Python垃圾回收机制清理。

详细分析Python垃圾回收机制

效率问题解决方案-分代回收

基于引用计数的垃圾回收机制每一次执行清理操作前都会将整个堆区的变量值的引用计数做一次遍历统计。这样做是非常消耗时间的,所以Python垃圾回收机制为了效率的提升加入了分代回收的策略。

详细分析Python垃圾回收机制

参考文献

https://www.jb51.net/article/161474.htm

来源:https://www.cnblogs.com/Yunya-Cnblogs/p/12805512.html

标签:Python,垃圾回收,机制
0
投稿

猜你喜欢

  • MySQL 数据库语句优化的原则

    2010-01-20 10:11:00
  • YOLOv5目标检测之anchor设定

    2022-04-23 16:22:10
  • c++基础语法:虚继承

    2024-01-23 10:01:00
  • 跟我学习javascript的闭包

    2024-04-23 09:11:51
  • Python数学形态学实例分析

    2022-11-22 22:32:50
  • Python多分支if语句的使用

    2022-07-17 17:52:54
  • python 接口_从协议到抽象基类详解

    2021-06-18 02:22:06
  • PyQt5中向单元格添加控件的方法示例

    2023-10-20 05:08:48
  • 高考要来啦!用Python爬取历年高考数据并分析

    2021-06-10 04:08:25
  • Python使用pycharm导入pymysql教程

    2024-01-17 22:47:49
  • Jenkins定时构建语法规则及时间设置

    2022-05-03 19:36:24
  • vue中element-ui组件默认css样式修改的四种方式

    2024-05-09 10:50:45
  • Python+Flask编写一个简单的行人检测API

    2023-09-26 17:55:19
  • VSCode + WSL 2 + Ruby环境搭建图文详解

    2022-09-10 21:05:38
  • js判断传入时间和当前时间大小实例(超简单)

    2024-05-02 17:26:40
  • 详解Python 装饰器执行顺序迷思

    2023-12-30 23:55:23
  • 详解Python手写数字识别模型的构建与使用

    2023-10-21 18:34:12
  • 使用python flask框架开发图片上传接口的案例详解

    2021-12-26 05:54:30
  • python鼠标绘图附代码

    2021-09-09 17:10:26
  • 如何爬取通过ajax加载数据的网站

    2022-05-03 15:06:01
  • asp之家 网络编程 m.aspxhome.com