我对终结器方法__del__
产生了疑问
在清理对象之前调用__del__
方法,但此方法甚至可以恢复对象。现在我注意到在Python 2.7中终结器被称为每次对象的引用计数器降为零(即使对象已经复活),在Python3.4中它被称为< em>一次对象的整个生命周期,所以如果它被复活然后它的计数器降到零,则不再调用终结器方法。
从PEP 442我在python 3.4中读到这个选择是为了阻止复活僵尸:
但是,如果对象已经完成,则不会调用终结器。这阻止我们完成僵尸
我怀疑的是,我无法弄清楚这个解决方案如何阻止我们完成僵尸,你能告诉我一些具体的例子吗?此外,该对象只能复活一次。
答案 0 :(得分:1)
假设我们有以下类:
class RefCycle:
def __init__(self):
self.refcycle = self
def __del__(self):
print('finalizing')
print(self.refcycle)
我们创建并收集一个循环隔离物:
RefCycle()
import gc
gc.collect()
当垃圾收集器检测到循环隔离时,它会调用RefCycle
实例的终结器。然后,它开始打破引用。当它清除self.refcycle
属性时,对象的引用计数将降为0.
这是最终确定的对象不被重新化的重要地方。如果那个“已经完成”的标志不存在,那么该对象将因refcount变为0而被重新归结。由于垃圾收集器已经弄乱了对象的属性,这将是一件非常糟糕的事情。 “已经最终确定”的标志保护我们免受这种情况的影响。
答案 1 :(得分:0)
我想我找到了一个理由,但如果它不是正确的话,请帮助我。
在PEP 442之后,CI的处理传统上是这样的:
新提案将成为:
因为你可以看到前者的第2步和后者的第4步是相同的,特别是它们现在使用相同的函数tp_clear
(并且这里帮助我因为我没有&t; t阅读代码,我不太好读C代码)这个函数,从我在销毁一个对象调用它的终结器之前推断出来的,但因为它已经被调用并且CI可能已经开始CT并且复活一个僵尸将是一个错误,终结器只调用一次。