Python 垃圾回收和弱引用(Weakref)

Python中的赋值语句是建立变量名与对象的引用关系,多个变量可以引用同一个对象,当对象的引用数归零时,可能会被当作垃圾回收。而弱引用即可以引用对象,又不会阻止对象被当作垃圾回收,因此这个特性非常适合用在缓存场景,当对象被当作垃圾回收时,其缓存信息会同步清除。

文章目录

  • 一、对象引用与垃圾回收
  • 二、弱引用/weakref模块
    • 2.1 weakref.ref 弱引用函数
    • 2.2 weakref.proxy 弱引用代理
    • 2.3 weakref.WeakValueDictionary 弱引用字典
    • 2.4 weakref.finalize 终结器
  • 三、弱引用的局限性

一、对象引用与垃圾回收

Python中的赋值语句不是创造对象而是建立引用关系,执行一个赋值语句时,如果对象不存在,则会先创建对象,然后建立变量与对象的引用关系。

下面赋值语句中,Python会先创建集合对象,让后建立变量名set1与对象的引用关系:

set1 = {1, 2, 3}

在这里插入图片描述

此时执行 set2 = set1,不会建立新的对象,而是建立变量名set2到集合对象的引用关系:
在这里插入图片描述

同样,del语句不是删除对象,而是删除引用关系,del set1只是删除了set1到集合对象的引用,而set2的引用依然存在,所以对象并不会被销毁:

del set1

在这里插入图片描述
每个对象都会统计有多少个引用指向自己,正是因为有引用对象才会存在。当对象的引用计数归零后,垃圾回收程序会将对象销毁并释放内存。

二、弱引用/weakref模块

所谓弱引用就是引用对象但是不增加引用计数,即弱引用关系不会妨碍对象被当做垃圾回收,但在实际销毁对象前,弱引用也能返回该对象。这种特性经常用在缓存场景,当对象在程序的其他地方被当垃圾回收后,缓存中对象也会自动删除。python中的weakref模块可以创建对象的弱引用。

2.1 weakref.ref 弱引用函数

weak.ref(object[, callback])创建对象的弱引用。可以通过调用weakref.ref的返回对象来获取其引用的内容,如果弱引用的对象还存在则返回引用对象,若不存在则返回None。若提供了回调函数callback,则在对象即将销毁时将调用回调函数,引用对象将作为回调函数的唯一参数,随后对象销毁。

示例:建立变量set1到集合{1, 2, 3}的弱引用,同时自定回调函数func,在对象销毁时打印"Object is gone!"提示我们:

def func(self):    #  自定义回调函数print("Object is gone!")import weakref
set1 = {1, 2, 3}
wref = weakref.ref(set1, func)    # 建立弱引用关系

在这里插入图片描述

调用wref()可以获得其引用的对象:

wref()   

在这里插入图片描述

将变量名set1引用至其他对象,原集合对象引用计数归零将被销毁,同时打印了提示信息。通过再次调用wref()可以确定其引用对象已被销毁(返回None):

set = 1
wref()

在这里插入图片描述

2.2 weakref.proxy 弱引用代理

weakref.proxy(object[, callback])创建一个弱引用的对象代理,代理不需要调用即可访问原对象内容:

import weakref
set1 = {1, 2, 3}
wref = weakref.proxy(set1) 

在这里插入图片描述

2.3 weakref.WeakValueDictionary 弱引用字典

WeakValueDictionary 是一个字典类,里面的值是对象的弱引用。当被引用的对象在程序的其他地方被当作垃圾回收后,对应的键会自动从 WeakValueDictionary 中删除,因此非常适合用于缓存。

示例:假设程序中定义了一个类person用于存储姓名和工资信息,details保存了该类的2个实例:

class person:def __init__(self,name, salary):self.name = nameself.salary = salarydef __repr__(self):return 'Name:{} => Salary:{}.format(self.name, self.salary)'details = [person('Vincent', 1000), person('Victor', 2000)]

在这里插入图片描述

在缓存中,可以定义一个弱引用字典来保存details中的内容:

weakdetails = weakref.WeakValueDictionary()
for detail in details:weakdetails[detail.name] = detail    # 建立实例和名称的弱引用

在这里插入图片描述

通过weakdetails.keys()可以看到其也引用了2个实例,假设程序中执行了del details[0] 将vincent的实例删除,再次调用weakdetails.keys()即可看到其中的键也自动删除了:

sorted(weakdetails.keys())
del details[0]    # 删除detials中的vincent实例
sorted(weakdetails.keys())    # weakdetails中的引用关系也同步删除了

在这里插入图片描述
和WeakValueDictionary对应的还有一个WeakKeyDictionary,其是键的弱引用,效果和上面类似,这里就不演示了。

2.4 weakref.finalize 终结器

weakref.finalize(obj, func, *arg, **args)返回一个终结器对象,即一个回调函数,在销毁对象时会被调用。使用finalize的主要好处是它能更简便的注册回调函数,而无需保留所返回的终结器对象。

a = {1,2,3}
weakref.finalize(a, print, 'Object a is gone!')    # 将终结器注册到变量a的对象上
a = 2    # 原集合被销毁,触发终结器回调函数

在这里插入图片描述

终结器对象在调用前都被视为存活状态(可通过.alive属性查询)。你也可以主动调用终结器,调用一次后则其死亡,对象销毁时只会调用存活状态的终结器(因此主动调用过的终结器不会触发):

b = {4,5,6}
f = weakref.finalize(b, print, 'Object b is gone!')    # 注册终结器对象
f.alive    # 查询终结器存活状态
f()    # 显式调用终结器
f.alive   # 调用后状态变为死亡
b = 7     # 其注册对象销毁,但不会调用已死亡的终结器

在这里插入图片描述

三、弱引用的局限性

弱引用的应用对象存在局限性,并不是所有Python对象都可以作为弱引目标。例如列表(list),字典(dict),整型(int),元组(tuple)对象不能作为弱引用的目标。

下面对列表对象创建弱引用,显示无法创建弱引用:

list1 = [1, 2, 3]
wref = weakref.ref(list1)

在这里插入图片描述

列表和字典可以通过创建子类来绕过此限制,下面Mylist为list的子类,其实例就可以被弱引用:

class Mylist(list):pass
list1 = Mylist([1, 2, 3])
wref = weakref.ref(list1)

在这里插入图片描述

但是对于int和tuple类型,即使通过子类,也无法建立弱引用:

class Myint(int):pass
a = Myint(1)
wref = weakref.ref(a)

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/577303.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

报错 /core/library/think/cache/driver/File.php 第 126 行左右(已解决)

报错 /core/library/think/cache/driver/File.php 第 126 行左右 解决方法: 网站后台版本低于v1.5.2出现的缓存问题,如果无法登录后台了,就通过FTP,把 /data/runtime 里的都删掉,然后进后台升级到最新版 一、进入宝…

vmWare虚拟机下载安装详细教程,手把手一步一步教学

一.下载 先去官网下载vm 官网地址:https://www.vmware.com/ 官网下载速度太慢了 国内应用商店或者别的下载地址:vmware下载_vmware下载免费中文版客户端[虚拟机]-下载之家 然后再去下载一个镜像 地址是:Index of /centos/7/isos/x86_64/ 我个人下载的是这个 ,看各位的需求…

【Java】ArrayList数组的扩容机制 jdk1.8

📝个人主页:哈__ 期待您的关注 ArrayList和普通数组不同,ArrayList支持动态扩容,那么ArrayList到底是如何扩容的呢?你又是否知道ArrayList数组的初始长度是多少呢? 在开始介绍之前,我们要先介…

代码随想录阅读笔记-二叉树【翻转二叉树】

题目 翻转一棵二叉树。 思路 如果要从整个树来看,翻转还真的挺复杂,整个树以中间分割线进行翻转,如图: 可以发现想要翻转它,其实就把每一个节点的左右孩子交换一下就可以了。 关键在于遍历顺序,前中后序应…

2024年妈妈杯数学建模思路A题B题C题D题思路分享

文章目录 1 赛题思路2 比赛日期和时间3 组织机构4 建模常见问题类型4.1 分类问题4.2 优化问题4.3 预测问题4.4 评价问题 5 建模资料 1 赛题思路 (赛题出来以后第一时间在CSDN分享) https://blog.csdn.net/dc_sinor?typeblog 2 比赛日期和时间 报名截止时间:2024…

​马来语翻译中文去哪比较好?

据了解,马来语是马来西亚、文莱的官方语言,也是新加坡的官方语言之一,马来语跟印尼语是同一种语言,它在整个东南亚有着极大的影响力。如今在国内市场上,马来语翻译的需求也是供不应求,那么,如何…

信息安全之网络安全防护

先来看看计算机网络通信面临的威胁: 截获——从网络上窃听他人的通信内容中断——有意中断他人在网络上的通信篡改——故意篡改网络上传送的报文伪造——伪造信息在网络上传送 截获信息的攻击称为被动攻击,而更改信息和拒绝用户使用资源的攻击称为主动…

利用Python和IP技术实现智能旅游情报系统

文章目录 引言一、系统架构设计1. 数据采集模块2. 数据处理模块3. 用户界面模块 二、数据获取技术应用三、系统功能展示四、亮数据采集工具介绍五、总结六、号外 引言 随着旅游行业的不断发展,人们对旅游信息的需求也越来越大。为了帮助旅行者更好地规划行程&#…

智慧酒店(二):AI智能分析网关V4视频分析技术在酒店管理中的应用

一、人工智能技术如何应用在酒店管理中? 随着科技的飞速发展,人工智能技术已经逐渐渗透到我们生活的方方面面,其中,酒店管理行业便是其应用的重要领域之一。人工智能技术以其高效、精准的特点,为酒店管理带来了革命性…

前端超分辨率技术应用:图像质量提升与场景实践探索-设计篇

超分辨率! 引言 在数字化时代,图像质量对于用户体验的重要性不言而喻。随着显示技术的飞速发展,尤其是移动终端视网膜屏幕的广泛应用,用户对高分辨率、高质量图像的需求日益增长。然而,受限于网络流量、存储空间和图像…

【报错解决】RuntimeError: CUDA error: out of memory(已解决)

【报错解决】RuntimeError: CUDA error: out of memory(已解决) 【先赞后看养成习惯】求关注点赞收藏 问题描述:在运行python代码中,遇到如下报错 return t.to(device, dtype if t.is_floating_point() or t.is_complex() else N…

153 Linux C++ 通讯架构实战8 ,日志打印实战,设置时区,main函数中顺序调整

日志打印实战 //日志的重要性:供日后运行维护人员去查看、定位和解决问题; //新文件:ngx_printf.cxx以及ngx_log.cxx。 //ngx_printf.cxx:放和打印格式相关的函数; //ngx_log.cxx:放和日志相关…