python 中 global 作用和全局变量实现方式

本文深入探讨了Python中全局变量的使用及潜在问题,通过具体示例解释了函数globals属性与global语句的工作原理,强调了避免全局变量以增强代码可读性和调试性的必要性。

摘要

  • global 标志实际上是为了提示 python 解释器,表明被其修饰的变量是全局变量。这样解释器就可以从当前空间 (currentscope) 中读写相应变量了。

  • Python 的全局变量是模块 (module) 级别的

  • 每个 python 函数拥有对应的 globals 字典,该字典与函数所属模块的 __dict__字典完全相同。函数的全局变量也会从这个字典中获取

    注:上面三句话的意思就是,python 解释器发现函数中的某个变量被 global 关键字修饰,就去函数的 __globals__字典变量中寻找(因为 python 中函数也是一等对象);同时,一个模块中每个函数的 globals 字典变量都是模块 dict 字典变量的引用,二者值完全相同。

  • 避免全局变量将使得程序更容易被调试,同时也能提升程序的可读性

动机

我最近遇到了一个关于 python 全局变量的问题,如下面这个简单例子里展示(当然实际代码要比这个复杂的多,这里只是一个抽象出来当例子)。例子中 foo.py 定义了函数 f,而函数 f 调用了全局变量 a:

# foo.py

def f():
    print(a)

def main():
    global a
    a = 5
    f()

if __name__ == '__main__':
    main()

运行上面这个文件将如预料中的输出5。在另一个文件 bar.py 中我们引入上面的 f,代码如下

# bar.py
from foo import f

def main():
    f()

main()

运行 bar.py 将报 NameError 错误。这是因为 a 被定义在 foo.py 的 main 函数中,而当导入 f 函数时, foo.py 的 main 函数并未被运行,所以 a 也没哟被定义。

Traceback (most recent call last):
  File "bar.py", line 10, in <module>
    main()
  File "bar.py", line 7, in main
    f()
  File "foo.py", line 5, in f
    print(a)
NameError: global name 'a' is not defined

定义全局变量 a

为了修复上面当问题第一反应是在 bar.py 中定义全局变量 a,这样 f 就可以找到变量 a 了,如下面的代码:

# bar.py
from foo import f

def main():
    global a
    a = 4
    f()

main()

然而依旧会报错,黑人问号脸???

Traceback (most recent call last):
  File "/tmp/example/bar.py", line 13, in <module>
    main()
  File "/tmp/example/bar.py", line 9, in main
    f()
  File "/tmp/example/foo.py", line 5, in f
    print(a)
NameError: global name 'a' is not defined

函数的 globals 属性与 python 的 global 语句

python 的 global 语句的作用只是提示 python 解释器,被 global 修饰的变量是一个全局变量,利用上面例子里函数 f 的反编译代码可以清除的看到这一点:

import dis
from foo import f

dis.dis(f)

5           0 LOAD_GLOBAL              0 (print)
            2 LOAD_GLOBAL              1 (a)
            4 CALL_FUNCTION            1
            6 POP_TOP
            8 LOAD_CONST               0 (None)
           10 RETURN_VALUE

从上面可以看出变量 a 被认为是全局变量。Python 中的每一个函数都拥有一个 globals 字典变量,该变量实际是函数所属模块的 dict 变量的引用。所以在 bar.py 中我们想在 bar.main 函数中将全局变量 a 赋值为4,实际改变的是 bar.py 的 dict 字典变量 (注:而不是定义 f 的 foo.py 的 dict 字典变量)

# bar.py
def main():
    global a
    a = 4
    print(main.__globals__.keys())
    print(main.__globals__['a'])
dict_keys(['__name__', '__doc__', '__package__', '__loader__', '__spec__', '__annotations__', '__builtins__', '__file__', '__cached__', 'foo', 'f', 'dis', 'main', 'a'])
4

上面的代码输出了 main.globals (即 bar.dict ) 中全局变量 a 的值是4,然而这个值对函数 f 来说确是不可见的,因为 f.globals 实际等于 foo.dict (简单而言就是命名空间不同)

from foo import f
print(f.__globals__)

假设我们在 foo.py 所有函数的外部预先定义了全局变量 a ,那么在将函数 f 导入时,a 会随着 f.globals 一同被导入。但这时被导入的 f.globals[“a”] ( 即 foo.dict[“a”] ) 和 bar.main 中赋值的 bar.main.globals[“a”] ( 即 bar.dict[“a”] ) 仍然不是同一个变量,即赋值无法改变函数 f 的输出,如下面的例子所示。

# foo.py
a = 3

def f():
    print(a)

def main():
    global a
    a = 5
    f()

if __name__ == '__main__':
    main()

# bar.py
from foo import f

def main():
    global a
    a = 4
    f()

main()

运行 bar.py 输出3,而不是 4。

就上述例子而言,如果我们想在 bar.py 中改变函数 f 的输出,则需要直接更新其 globals 变量的值。

# bar.py
from foo import f

def main():
    f.__globals__['a'] = 4
    f()

main()

模块的 dict 变量和猴子布丁 (monkey-patching)

如上所述,函数的 globals 变量实际是其所属模块 dict 变量的引用。所以为了达到上面修改全局变量的目的,也可以直接更新 foo.dict 。修改模块 foo 的属性 (attribute) 值即可直接更新 foo.dict

# bar.py
import foo
from foo import f


def main():
    foo.a = 4
    f()

如果你曾经使用过运行中给代码打补丁的库,一般就是这么实现的。直接修改被打补丁的模块的 dict 中特定的对象或函数。

输入使得函数变得更加容易测试

上面的例子中的函数 f 如果接受输入变量的话,而不是使用全局变量,代码将更容易被测试。同时可读性也更好,出了问题也更容易 debug。

# foo.py
def f(a):
    print(a)


def main():
    a = 5
    f(a)

if __name__ == '__main__':
    main()

# bar.py
from foo import f

def main():
    a = 3
    f(a)

<think>嗯,用户现在询问的是Pythonglobal关键字用法,重点在确认它是否用于定义全局变量。结合用户前面提到过的“globalCPFSU”(虽然CPFSU的具体含义未明,但可以聚焦当前问题),我需要根据最新提供的Python引用内容来解答。用户给的引用内容都很明确:引用[1][2]直接说明global用于在函数内部修改全局变量,引用[3][4]强调了变量作用域规则,引用[5]的示例则展示了全局列表的内存地址不变性。这说明用户需要的是准确的技术解释而非猜测性内容。注意到用户特别关注“定义”“修改”的区别,这点很关键。从专业角度看,Pythonglobal实际是作用域声明指令而非定义操作——它不创建变量只绑定已有变量。引用[2]的示例最能说明这点:函数内部globalx只是将局部x关联到外部x,真正的变量定义在函数外。另外用户可能隐含深层疑问:为什么需要global而不是直接用外部变量?这就要解释Python的LEGB作用域规则(根据引用[3][4]),函数内部默认不能修改全局层变量。用引用[5]的可变类型示例会更直观——当变量是列表时,即便不用global也能修改元素(引用[5]展示的test02情况很说明问题),但重新赋值就需要global。最终框架会分三块:基本用法(区分定义/声明)、必要场景说明、可变对象特例。尤其要强调“修改=赋值操作”这个易错点——很多初级错误源于误解赋值原地操作的区别。引用[5]的test01test02函数对比恰好能形象解释这个概念。</think> 在 Python 中,`global`关键字**不用于定义全局变量**,而是用于声明函数内部对**已有全局变量的修改权限**。以下是核心解析: --- ### 一、`global`的核心用途 #### 1. **声明全局变量修改权** - 当函数内部需要修改**函数外部定义的全局变量**时,必须用`global`声明该变量。 - *未声明时*:函数内部对变量赋值会创建新**局部变量**,而非修改全局变量(可能导致逻辑错误[^4])。 *示例代码(错误示范):* ```python x = 10 # 全局变量 def func(): x = 20 # 创建局部变量x,而非修改全局x func() print(x) # 输出: 10(全局x未被修改) ``` #### 2. **正确修改全局变量的步骤** ```python x = 10 # 全局变量定义在函数外部 def modify_global(): global x # 声明x为全局变量 x = 20 # 修改全局x modify_global() print(x) # 输出: 20 [^2][^5] ``` - **`global x`仅声明权限**,全局变量本身需在函数外定义。 --- ### 二、关键场景与规则 #### 1. **修改不可变对象必用`global`** - 若全局变量为**不可变对象**(如整数、字符串),函数内部修改必须使用`global`: ```python count = 0 def increment(): global count count += 1 # 修改整数需声明 ``` #### 2. **可变对象的特殊规则** - 若全局变量为**可变对象**(如列表、字典),直接修改其内容(如`append()`)**无需**`global`: ```python test_list = [1, 2] # 全局列表 def modify_list(): test_list.append(3) # 无需global,直接操作元素 modify_list() print(test_list) # 输出: [1, 2, 3] [^5] ``` - 但若对可变对象**重新赋值**(如`test_list = [4, 5]`),仍需`global`声明。 --- ### 三、总结与最佳实践 | **场景** | 是否需`global` | 说明 | |-------------------------|---------------|-----------------------------| | 读取全局变量 | ❌ 不需要 | 函数可直接访问 | | 修改不可变全局变量 | ✔️ 必需 | 如`int`, `str`, `tuple` | | 修改可变全局变量的内容 | ❌ 不需要 | 如`list.append()`, `dict.update()` | | 对可变全局变量重新赋值 | ✔️ 必需 | 如`list = [新列表]` | > **最佳实践**: > 1. 避免滥用全局变量,优先用函数参数返回值传递数据[^3]; > 2. 必须修改全局变量时,用`global`显式声明提高代码可读性。 --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值