我在尝试删除对象的最后一个引用时遇到了一个奇怪的问题。
代码是:
import sys
import weakref
class Ref:
def __init__(self, name):
self.name = name
self.ref = []
def reference(self, obj):
name = obj.name
self.ref.append(
weakref.ref(obj, lambda wref: print('{!r} is dead.'.format(name))))
a = Ref('a')
b = Ref('b')
c = Ref('c')
d = Ref('d')
a.reference(b)
b.reference(c)
c.reference(d)
d.reference(a)
print('reference count before killed:', sys.getrefcount(d.ref[0]()))
del a
print('reference count after killed:', sys.getrefcount(d.ref[0]()))
输出是这样的:
reference count before killed: 2
'a' is dead.
reference count after killed: 1547
'd' is dead.
'c' is dead.
但有时候(它完全是随机的)我只得到'd' is dead.
或'c' is dead.
(但从不'b' is dead.
),或根本没有这些消息。
所以我的第一个问题是:这个奇怪的引用计数是什么1547
?它来自哪里?
第二个是:为什么杀死实例a
会创建这个随机的“杀死其他实例”效果?
答案 0 :(得分:2)
GC a
后,d.ref[0]()
生成None
。这就是为什么你在删除a
后得到1547的引用数量的原因;毕竟,你不能要求收集对象的引用,是吗?
c
和d
的奇怪删除是因为Python不保证解释器退出时对象是否存活将经历正常的销毁过程。在您的脚本结束时,b
,c
和d
都处于活动状态。有时候,他们会正常得到GC,弱反复回调会运行。有时候,这种情况不会发生。 Python在这里没有做出任何承诺。