任何修补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中使用了静态变量,而没有滋生bug?)
编辑
:
Baczek made an interesting example
. 以及您的大多数评论和
Utaal's in particular
,我进一步阐述:
>>> def a():
... print("a executed")
... return []
...
>>>
>>> def b(x=a()):
... x.append(5)
... print(x)
...
a executed
>>> b()
[5]
>>> b()
[5, 5]
对我来说,设计决策似乎与参数的范围放在哪里有关:在函数内部,还是与它“一起”?
在函数内部进行绑定意味着
x
当函数被调用(而不是定义)时,有效地绑定到指定的默认值,这可能会带来一个严重的缺陷:
def
行是“混合”的,因为部分绑定(函数对象)将在定义时发生,部分绑定(默认参数的分配)将在函数调用时发生。
实际行为更加一致:当执行该行时,该行的所有内容都会得到评估,这意味着在函数定义时。