任何对 Python 进行过长期修改的人都会被下面的问题所困扰(或撕碎):def foo(a=[]): a.append(5) return aPython 新手会期望这个函数名为 wi...
任何长期使用 Python 的人都曾被下面的问题困扰过(或者说被折磨得遍体鳞伤):
def foo(a=[]):
a.append(5)
return a
Python 新手会认为这个不带参数的函数总是返回一个只有一个元素的列表: [5]
。但结果却大不相同,而且非常令人惊讶(对于新手来说):
>>> foo()
[5]
>>> foo()
[5, 5]
>>> foo()
[5, 5, 5]
>>> foo()
[5, 5, 5, 5]
>>> foo()
我的一位经理曾经第一次遇到这个特性,并称其为该语言的“重大设计缺陷”。我回答说,这种行为有其根本原因,如果你不了解内部原理,这确实非常令人费解和意外。然而,我无法回答(对自己)以下问题:在函数定义而不是在函数执行时绑定默认参数的原因是什么?我怀疑这种经验行为是否有实际用途(谁真的在 C 中使用了静态变量而不会产生错误?)
编辑 :
Baczek 举了一个有趣的例子 。结合你们的大多数评论, 尤其是 Utaal 的评论 ,我进一步阐述:
def a():
print("a executed")
return []
def b(x=a()):
x.append(5)
print(x)
a executed
>>> b()
[5]
>>> b()
[5, 5]
对我来说,设计决策似乎与将参数范围放在哪里有关:在函数内部,还是与函数“一起”?
在函数内部进行绑定意味着 x
在调用函数时有效地绑定到指定的默认值,而不是定义函数,这将带来一个严重的缺陷:该 def
行将是\'混合\',因为部分绑定(函数对象)将在定义时发生,而部分(默认参数的分配)将在函数调用时发生。
实际行为更加一致:当执行该行时,该行的所有内容都会得到评估,即在函数定义时。