代码之家  ›  专栏  ›  技术社区  ›  olinox14

为什么作为可选参数传递的列表的行为是这样的?[副本]

  •  -1
  • olinox14  · 技术社区  · 5 年前

    任何修补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 行是“混合”的,因为部分绑定(函数对象)将在定义时发生,部分绑定(默认参数的分配)将在函数调用时发生。

    实际行为更加一致:当执行该行时,该行的所有内容都会得到评估,这意味着在函数定义时。

    0 回复  |  直到 2 年前