Python 对象引用循环如何产生

1次阅读

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

Python 对象引用循环如何产生

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.WeakKeyDictionaryWeakValueDictionary 适合缓存场景
  • 在上面的父子节点例子中,让 parent 属性使用 weakref.ref 就能打破循环

text=ZqhQzanResources