Python 列表推导式中的名称重新绑定
列表推导式提供了一种在 Python 中生成列表的优雅方法。然而,它们表现出特殊的作用域语义,可能会导致意外的行为。
考虑以下代码片段:
x = "original value" squares = [x**2 for x in range(5)] print(x) # Prints 4 in Python 2!
令人惊讶的是,在 Python 2 中,此代码将打印“4”而不是“原值。”发生这种情况是因为列表推导将循环控制变量(此处为“x”)“泄漏”到周围范围中。这意味着任何后续对“x”的引用都将引用推导式中分配的值,即使推导式之外存在同名变量也是如此。
此行为继承自列表推导式的原始实现,它优先考虑性能而不是封装。在Python 3中,这个“肮脏的小秘密”被消除了。列表推导式现在使用与生成器表达式相同的实现策略,可以防止名称隐藏。因此,在 Python 3 中,上述代码将打印“原始值”。
Python 的创建者 Guido van Rossum 解释了这一更改背后的原理:
“在 Python 中2、列表推导式将循环控制变量泄漏到周围范围中,这是无意的,并导致了混乱和错误,在 Python 3 中,此行为已更改为匹配生成器表达式,从而阻止了名称。
虽然此更改增强了 Python 代码的稳健性,但了解 Python 2 和 3 之间的不同行为至关重要。正确的封装技术(例如在推导式中使用下划线作为循环控制变量前缀)可以帮助减轻这种潜在的陷阱。
以上是Python 2 和 3 列表推导式中的名称重新绑定有何不同?的详细内容。更多信息请关注PHP中文网其他相关文章!