python对象引用循环指多个对象互相持有对方引用致引用计数无法归零,如父子节点互引;容器自引用、闭包嵌套、自定义__del__方法及弱引用使用不当均会加剧该问题,weakref可主动避免。

Python 中的对象引用循环,通常发生在两个或多个对象互相持有对方的引用,导致它们的引用计数无法降为 0,从而在常规引用计数机制下无法被立即回收。
常见产生场景
最典型的是两个实例对象相互赋值属性:
- 对象 A 的某个属性指向对象 B
- 对象 B 的某个属性又指向对象 A
- 此时即使外部所有变量都不再引用 A 和 B,它们仍彼此“拽着”对方
例如:
class Node: def __init__(self, name): self.name = name self.parent = None self.children = [] <p>a = Node("a") b = Node("b") a.children.append(b) # a → b b.parent = a # b → a</p><h1>此时 a 和 b 构成引用循环</h1>
容器类型加剧循环风险
列表、字典、集合等容器对象容易隐式形成循环:
立即学习“Python免费学习笔记(深入)”;
- 一个列表把自己作为自身某个元素(
lst.append(lst)) - 字典的键或值引用了包含它的对象(如类实例把自身存入自己的
__dict__) - 闭包中嵌套函数引用了外层作用域的可变对象,而该对象又持有函数引用
自定义 __del__ 方法可能诱发循环
当对象定义了 __del__ 方法,且该对象参与了循环引用时,CPython 的垃圾回收器会将它加入“不可达但带 __del__”的特殊队列,延迟处理甚至不处理,进一步阻碍清理。
- 这类对象不会被循环垃圾收集器自动释放,除非手动调用
gc.collect() - 若多个带
__del__的对象互引,还可能引发析构顺序不确定或异常抑制
弱引用可主动避免循环
对不需要“强持有”的关系,改用 weakref 模块:
-
weakref.ref(obj)创建弱引用,不增加引用计数 -
weakref.WeakKeyDictionary或WeakValueDictionary适合缓存场景 - 在上面的父子节点例子中,让
parent属性使用weakref.ref就能打破循环