“最小惊讶”和可变默认参数
任何修改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
实际行为更加一致:当执行该行时,该行的所有内容都会被评估,这意味着在函数定义时。