【问题标题】:NameError vs global NameErrorNameError 与全局 NameError
【发布时间】:2021-02-07 05:53:35
【问题描述】:

我正在尝试重现我遇到的显示此问题的错误:

NameError:未定义全局名称“sdrent”

但是,如果我打开解释器并输入 sdrent,则会收到以下错误:

>>> sdrent
NameError: name 'sdrent' is not defined

NameError: global name...NameError: name... 有什么区别,我该如何重现前者?

【问题讨论】:

标签: python python-2.7


【解决方案1】:

CPython 有两个用于全局变量查找的操作码,LOAD_GLOBALLOAD_NAMELOAD_NAME 在全局变量之前查找局部变量,而 LOAD_GLOBAL 直接查找全局变量。 LOAD_NAME 主要用于类语句,但在没有 global 声明的情况下,编译器也恰好发出 LOAD_NAME 用于模块级别的全局变量查找。

回到 Python 3.4 之前,LOAD_GLOBAL 曾经在查找失败时表示 global name 'whatever' is not defined,而 LOAD_NAME 曾经表示 name 'whatever' is not defined。当有人争辩说“全局”在有人输入错误的局部变量名称的情况下令人困惑时,这得到了changed

您使用的是 Python 2.7。当您在顶层运行一个不存在的名称的变量查找时,您会收到 LOAD_NAME 错误消息,但在函数内部,您会收到 LOAD_GLOBAL 错误消息,在 Python 2 上仍然显示“全局”。

【讨论】:

    【解决方案2】:

    这似乎发生在函数或方法的上下文中,其中 LEGB 停止在 G(lobal),知道内置函数并在该范围内退出/抱怨。

    例如,重现:

    >>> def hi():
    ...     sdrent
    ...
    >>> hi()
    NameError: global name 'sdrent' is not defined
    

    甚至更简单:

    >>> (lambda: sdrent)()
    NameError: global name 'sdrent' is not defined
    

    在类方法中:

    >>> class X:
            def __call__(_): sdrent
    >>> X()()
    NameError: global name 'sdrent' is not defined
    

    【讨论】:

    • 这与“已经知道内置”没有任何关系。查找实际上会继续到内置插件,因为即使它试图缓存没有内置 sdrent 的事实,同时也可能会在内置插件中添加 sdrent
    • @user2357112supportsMonica 我明白了,也许我应该添加它以澄清它确实检查了这一点,尽管错误消息没有提到这一点。
    • (在较新的 Python 版本上实际上有一个全局/内置变量缓存优化,基于 dict 版本标识符,但它只在查找成功的情况下触发,而不是 NameErrors。)
    猜你喜欢
    • 1970-01-01
    • 2011-04-27
    • 1970-01-01
    • 2014-10-24
    • 2012-05-29
    • 2013-08-23
    • 2014-04-03
    • 2017-05-12
    相关资源
    最近更新 更多