当前位置: 首页 > 面试题库 >

我不了解python __del__的行为

丁鸿云
2023-03-14
问题内容

有人可以解释为什么以下代码会如此行为:

import types

class Dummy():
    def __init__(self, name):
        self.name = name
    def __del__(self):
        print "delete",self.name

d1 = Dummy("d1")
del d1
d1 = None
print "after d1"

d2 = Dummy("d2")
def func(self):
    print "func called"
d2.func = types.MethodType(func, d2)
d2.func()
del d2
d2 = None
print "after d2"

d3 = Dummy("d3")
def func(self):
    print "func called"
d3.func = types.MethodType(func, d3)
d3.func()
d3.func = None
del d3
d3 = None
print "after d3"

输出(请注意,永远不会调用d2的析构函数)是(python 2.7)

delete d1
after d1
func called
after d2
func called
delete d3
after d3

有没有一种方法可以“修复”代码,以便在不删除添加的方法的情况下调用析构函数?我的意思是,放置d2.func = None的最佳位置将在析构函数中!

谢谢

[edit]基于前几个答案,我想澄清一下,我并不是在问using的优点(或不足之处)__del__。我试图创建最短的函数来证明我认为是非直觉的行为。我假设已经创建了一个循环引用,但是我不确定为什么。如果可能的话,我想知道如何避免使用循环引用。


问题答案:

我提供自己的答案是因为,尽管我喜欢避免使用的建议__del__,但我的问题是如何使它在提供的代码示例中正常工作。

简短版本: 以下代码用于weakref避免循环引用。我以为在发布问题之前就已经尝试过了,但是我想我一定做错了什么。

import types, weakref

class Dummy():
    def __init__(self, name):
        self.name = name
    def __del__(self):
        print "delete",self.name

d2 = Dummy("d2")
def func(self):
    print "func called"
d2.func = types.MethodType(func, weakref.ref(d2)) #This works
#d2.func = func.__get__(weakref.ref(d2), Dummy) #This works too
d2.func()
del d2
d2 = None
print "after d2"

较长的版本 :发布问题时,我确实搜索了类似的问题。我知道您可以with改用,并且普遍的看法__del__BAD

使用with是有意义的,但仅在某些情况下才有意义。打开,读取和关闭文件with是一个很好的解决方案的好例子。您已经找到了需要该对象的特定代码块,并且想要清理该对象和该块的末尾。

似乎经常使用数据库连接作为一个使用效果不佳的示例with,因为您通常需要离开创建连接的代码部分,并在更多的事件驱动(而不是顺序)的时间范围内关闭连接。

如果with不是正确的解决方案,我会看到两种选择:

  1. 您确定__del__可以正常工作(有关弱引用用法的更多说明,请参见此博客)
  2. 您可以在atexit程序关闭时使用该模块运行回调。例如,请参阅本主题。

当我尝试提供简化的代码时,我的真正问题更多是由事件驱动的,因此with不是一个合适的解决方案(with适用于简化的代码)。我也想避免使用atexit,因为我的程序可以长时间运行,并且希望能够尽快执行清理。

因此,在这种特定情况下,我发现它是使用weakref并防止循环引用__del__无法正常工作的最佳解决方案。

这可能是该规则的例外,但是在某些情况下,使用weakref__del__是正确的实现,恕我直言。



 类似资料:
  • 问题内容: 恐怕我一定不明白弹性增长。如果您跳到下面的JSFiddle(据我所知),它的大小应该是另一个的三倍。如您所见,事实并非如此。为什么? 问题答案: 您还必须指定一个值(不指定此属性会导致行为类似于使用初始值auto 将其添加到两个孩子中,或仅使用速记进行设置:

  • 问题内容: 第一个是枚举类 这是类CoffeeTest1,主要 以下是输出 我的问题:我不了解如何管理输出8。我没有给出构造函数对象(例如:)。我想知道背后的这种简单的微妙逻辑。有人可以帮我理解吗? 问题答案: 我不明白“ drink1.size.getounces()”如何管理输出8。 […] 我想知道背后的这种简单的微妙逻辑。 要了解其背后的逻辑,您可以将您的代码视为常规代码(实际上就是它的编

  • 当我尝试运行我的世博会项目时,我收到以下信息: 如果我运行npm安装,我会得到以下结果: 这就是我运行时得到的结果: 这是因为我尝试更新expo sdk,但我不知道我做错了什么。有人能帮我吗?

  • 问题内容: 嗨,我是Ankur,并且已经使用Java进行编码了几年。.我早些时候读过Herbert Schildt撰写的Head first Java The Complete Reference,最近在Oracle文档的这一页中遇到了一个重大谬误。注释部分说,如果构造函数未显式调用超类构造函数,则Java编译器会自动将调用插入到超类的无参数构造函数中。如果超类没有无参数构造函数,则将出现编译时错

  • 问题内容: 我来自Java SE背景,并且做了一些servlet教程,并阅读了Head First JSP和servlet。我现在正在阅读有关异步支持的JavaWorld.com文章,但我不太了解。 异步到底是什么?Ajax和Servlet Async有什么区别? PS我有一个使用ajax的PHP背景,我知道这个概念,但是我还没有用java尝试过 问题答案: 在传统的Servlet模型中,通常1个