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

python-为什么要修改整个列表的第一项[重复]

  •  0
  • Steztric  · 技术社区  · 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中使用了静态变量,而没有滋生错误?)

    编辑 :

    巴克泽举了一个有趣的例子。连同你的大部分评论,特别是尤塔尔的评论,我进一步阐述了:

    >>> 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 回复  |  直到 5 年前
        1
  •  0
  •   Martijn Pieters    11 年前
    >>> def a():
    >>>    print "a executed"
    >>>    return []
    >>> x =a()
    a executed
    >>> def b(m=[]):
    >>>    m.append(5)
    >>>    print m
    >>> b(x)
    [5]
    >>> b(x)
    [5, 5]