python垃圾回收机制

本文详细介绍了Python的垃圾回收机制,包括引用计数、标记清除和分代回收。引用计数器通过跟踪对象的引用次数实现内存管理,但存在循环引用问题。为解决这一问题,Python引入了标记清除和分代回收策略。文章还讨论了Python中的缓存机制,如池和free_list,以提高性能和节省内存。
摘要由CSDN通过智能技术生成

python垃圾回收机制

现在的高级语言如java,c#等,都采用了垃圾收集机制,而不再是c,c++里用户自己管理维护内存的方式。自己管理内存极其自由,可以任意申请内存,但如同一把双刃剑,为大量内存泄露,悬空指针等bug埋下隐患。
对于一个字符串、列表、类甚至数值都是对象,且定位简单易用的语言,自然不会让用户去处理如何分配回收内存的问题。
python里也同java一样采用了垃圾收集机制,不过不一样的是:
python采用的是引用计数机制为主,标记-清除和分代收集(隔代回收)两种机制为辅的策略。

一、引用计数器

1.1环状的双向链表(Refchain)

图片

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

例如:

name = "小猪佩奇"   # 字符串对象
age = 18    # 整形对象
hobby = ["吸烟","喝酒","烫头"]   # 列表对象

这些对象都会放到这些双向链表当中,也就是帮忙维护了python中所有的对象。
也就是说如果你得到了refchain,也就得到了python程序中的所有对象。

1.2不同类型对象的存放形式

刚刚提到了所有的对象都存放在环状的双向链表中,而不同类型的对象存放在双向链表中既有一些共性特征也有一些不同特征。

# name = "小猪佩奇"   
# 创建这个对象时,内部会创建一些数据,并且打包在一起
# 哪些数据:【指向上一个对象的指针、指向下一个对象的指针、类型(这里为字符串)、引用的个数】 
"""
引用的个数:
    比如 name = '小猪佩奇' ,会给“小猪佩奇”开辟一个内存空间用来存放到双向链表中。
    这时候如果有 new = name,不会创建两个“小猪佩奇”,而是将new指向之前的那个小猪佩奇,
    而引用的个数变为2,也就是"小猪佩奇"这个对象被引用了两次。
"""
  • 相同点:刚刚讲到的四个种数据每个对象都包含有。
# 内部会创建一些数据,【指向上一个对象的指针、指向下一个对象的指针、类型、引用的个数】
age = 18    # 整形对象
# 内部会创建一些数据,【指向上一个对象的指针、指向下一个对象的指针、类型、引用的个数】
hobby = ["吸烟","喝酒","烫头"]   # 列表对象
  • 不同点:

不同的数据类型还会创建不同的值:

# 内部会创建一些数据,【指向上一个对象的指针、指向下一个对象的指针、类型、引用的个数、val=18】
age = 18    # 整形对象
# 内部会创建一些数据,【指向上一个对象的指针、指向下一个对象的指针、类型、引用的个数、items=元素、元素的个数】
hobby = ["抽烟","喝酒","烫头"]   # 列表对象

所以在python中创建的对象会加到环形双向链表中,但是每一种类型的数据对象在存到链表中时,所存放的数据个数可能是不同的(有相同点有不同点)。

两个重要的结构体

Python解释器由c语言开发完成,py中所有的操作最终都由底层的c语言来实现并完成,所以想要了解底层内存管理需要结合python源码来进行解释。

#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个值:_ob_next、_ob_prev、ob_refcnt、*ob_type)
9-13行 定义了一个结构体,第10行实际上就是6,7两行,用来存放前一个对象,和后一个对象的位置。

这个结构体可以存贮四个值(这四个值是对象都具有的)。

在C源码中如何体现由多个元素组成的对象:PyObject + ob_size(元素个数)

15-18行又定义了一个结构体,第16行相当于代指了9-13行中的四个数据。

而17行又多了一个数据字段,叫做元素个数,这个结构体。

以上源码是Python内存管理中的基石,其中包含了:

  • 2个结构体
  • PyObject,此结构体中包含3个元素。
    • PyObject_HEAD_EXTRA,用于构造双向链表。
    • ob_refcnt,引用计数器。
    • *ob_type,数据类型。
  • PyVarObject,次结构体中包含4个元素(ob_base中包含3个元素)
    • ob_base,PyObject结构体对象,即:包含PyObject结构体中的三个元素。
    • ob_size,内部元素个数。

类型封装的结构体

在我们了解了这两个结构体,现在我们来看看每一个数据类型都封装了哪些值:

  • flaot类型

float结构体:

typedef struct {
   
  
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值