删除最后一个引用后的奇怪删除行为

时间:2013-09-17 01:05:14

标签: oop python-3.x reference weak-references

我在尝试删除对象的最后一个引用时遇到了一个奇怪的问题。

代码是:

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会创建这个随机的“杀死其他实例”效果?

1 个答案:

答案 0 :(得分:2)

GC a后,d.ref[0]()生成None。这就是为什么你在删除a后得到1547的引用数量的原因;毕竟,你不能要求收集对象的引用,是吗?

cd的奇怪删除是因为Python不保证解释器退出时对象是否存活将经历正常的销毁过程。在您的脚本结束时,bcd都处于活动状态。有时候,他们会正常得到GC,弱反复回调会运行。有时候,这种情况不会发生。 Python在这里没有做出任何承诺。