这可能很愚蠢,但它已经在我的大脑后面唠叨了一段时间。
Python 为我们提供了两种从对象中删除属性的内置方法,del命令字和delattr内置函数。我更喜欢delattr,因为我认为它更明确一点:
del foo.bar
delattr(foo, "bar")
但我想知道它们之间是否存在内部差异。
第一个比第二个更有效。 del foo.bar
编译成两条字节码指令:
2 0 LOAD_FAST 0 (foo)
3 DELETE_ATTR 0 (bar)
而delattr(foo, "bar")
需要五个:
2 0 LOAD_GLOBAL 0 (delattr)
3 LOAD_FAST 0 (foo)
6 LOAD_CONST 1 ('bar')
9 CALL_FUNCTION 2
12 POP_TOP
这意味着第一次运行的速度稍快(但差别不大——在我的机器上为 0.15 μs)。
就像其他人所说的那样,当您要删除的属性是动态确定的时,您真的应该只使用第二种形式。
[编辑显示函数内部生成的字节码指令,编译器可以在其中使用LOAD_FAST
和LOAD_GLOBAL
]
考虑以下示例:
for name in ATTRIBUTES:
delattr(obj, name)
或者:
def _cleanup(self, name):
"""Do cleanup for an attribute"""
value = getattr(self, name)
self._pre_cleanup(name, value)
delattr(self, name)
self._post_cleanup(name, value)
你不能用del来做。
毫无疑问是前者。在我看来,这就像问是否foo.bar
比 更好getattr(foo, "bar")
,而且我认为没有人在问这个问题:)
这确实是一个偏好问题,但第一个可能更可取。如果您不知道要提前删除的属性的名称,我只会使用第二个。
就像 getattr 和 setattr 一样,delattr 只能在属性名称未知时使用。
从这个意义上说,它大致相当于用于访问比您通常可用的更低级别的内置功能的几个 python 功能,例如__import__
代替import
和operator.add
代替+
不确定内部工作原理,但从代码可重用性和不要成为混蛋同事的角度来看,使用 del。来自其他语言的人也更清楚和理解。
如果您认为delattr
更明确,那么为什么不getattr
一直使用而不是object.attr
?
至于引擎盖下......你的猜测和我的一样好。如果不是明显更好。
这是一个老问题,但我想投入我的 2 美分。
虽然,del foo.bar
更优雅,但有时你会需要delattr(foo, "bar")
. 比如说,如果您有一个交互式命令行界面,允许用户通过键入 name来动态删除对象中的任何成员,那么您别无选择,只能使用后一种形式。