在__del__中关闭/处理对象真的可以吗?


问题内容

我一直在思考如何用Python编写类。更具体地说,如何实现构造函数以及应如何销毁对象。我不想依靠CPython的引用计数来进行对象清理。这基本上告诉我,我应该使用with语句来管理对象的生命周期,并且需要一个显式的close
/ dispose方法(__exit__如果对象也是上下文管理器,则可以从该方法调用)。

class Foo(object):
    def __init__(self):
        pass
    def close(self):
        pass

现在,如果我所有的对象都以这种方式运行,并且我所有的代码都与语句或对close()(或dispose())的显式调用一起使用,那么我真的不觉得需要在其中放置任何代码__del__。我们是否应该真正使用__del__我们的物品?


问题答案:

简短答案:否。

长答案:使用起来__del__很棘手,主要是因为不能保证调用它。这意味着您不能在那里做绝对必须做的事情。反过来,这意味着__del__基本上只能用于无论如何迟早都会进行的清理,例如清理在进程退出时将被清理的资源,因此,是否__del__被调用无关紧要。当然,这些通常也与Python为您所做的相同。这样有点__del__没用。

另外,__del__当Python垃圾回收时,您会被调用,而您不想等待Python垃圾回收,这意味着您__del__无论如何都无法使用。

因此,请勿使用__del__。使用__enter__/__exit__代替。

仅供参考:这是一个非循环情况的示例,其中没有调用析构函数:

class A(object):
    def __init__(self):
        print('Constructing A')

    def __del__(self):
        print('Destructing A')

class B(object):
    a = A()

好的,这是一个类属性。显然,这是一个特例。但这只是表明确保__del__被调用并非易事。我敢肯定,我已经看到了更多非循环的情况,__del__没有被称为。