【问题标题】:Why does assigning to a class attribute with the same name as a local variable raise a NameError?为什么分配给与局部变量同名的类属性会引发 NameError?
【发布时间】:2018-08-18 14:42:13
【问题描述】:

考虑这个类定义:

def func():
    x = 5

    class Foo:
        x = x

func()

我希望这会创建一个属性 x 设置为 5 的类 - 但相反,它会引发 NameError:

Traceback (most recent call last):
  File "untitled.py", line 7, in <module>
    func()
  File "untitled.py", line 4, in func
    class Foo:
  File "untitled.py", line 5, in Foo
    x = x
NameError: name 'x' is not defined

但是,该错误仅在函数内部引发,并且仅当 x 是局部变量时。所有这些 sn-ps 都可以正常工作:

x = 5

class Foo:
    x = x
x = 5

def func():
    class Foo:
        x = x

func()
class Bar:
    x = 5

    class Foo:
        x = x
def func():
    x = 5

    class Foo:
        y = x

func()

是什么导致了这种奇怪的行为?

【问题讨论】:

  • 看起来像一个全球与非本地问题
  • 对于你想要的行为,只需在类定义之后设置Foo.x = x
  • This issue 似乎相关。

标签: python function class attributes


【解决方案1】:

因为 Python 编译器在确定哪些变量是本地变量时会查看代码块中的所有左值。由于在您有问题的代码中,x 被用作class 块中的左值,因此它被视为class 块的局部变量,因此,它被认为是在它被引用之前被引用赋值,因此异常。

【讨论】:

  • 那么这是一个 CPython 问题吗?你会说这是一个错误吗?
  • 我不会说这是一个错误。只是设计使每个变量的范围对于任何给定的代码块都是固定的,因此具有 x = x 的代码块中的变量 x 不必从父范围变量转换为局部变量运行。我同意这有点违反直觉,但为了提高效率,它可能会更好。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2021-01-12
  • 1970-01-01
  • 1970-01-01
  • 2018-12-11
  • 2012-09-10
  • 1970-01-01
相关资源
最近更新 更多