首页
学习
活动
专区
圈层
工具
发布
社区首页 >专栏 >一篇文章讲清楚Python的变量作用域

一篇文章讲清楚Python的变量作用域

原创
作者头像
风一样的男子
发布2026-06-11 11:22:24
发布2026-06-11 11:22:24
120
举报
文章被收录于专栏:编程教程编程教程

一个让我调试了2小时的bug

去年有个朋友发给我一段代码,让我帮忙看看问题出在哪。

他写了一个小程序,用来统计用户输入的数字:

代码语言:javascript
复制
total = 0

def process_numbers():
    while True:
        num = input("请输入数字(输入q退出):")
        if num == 'q':
            break
        total = total + int(num)
        print(f"当前总和:{total}")

process_numbers()
print(f"最终总和:{total}")

运行结果:

代码语言:javascript
复制
请输入数字(输入q退出):5
UnboundLocalError: local variable 'total' referenced before assignment

他懵了。total明明在函数外面定义好了,为什么报错说没定义?

他试了另一个写法:

代码语言:javascript
复制
count = 10

def show():
    print(count)

show()  # 输出10,正常运行

奇怪,这个就能读到。为什么一个能读,一个不能改?

那天晚上,我花了一个小时给他讲清楚什么是变量作用域。今天我把这些内容整理出来,希望帮你绕开同样的坑。


什么是作用域?一个快递站的比喻

先别急着看代码,我们来想一个生活中的场景。

你家住在一个小区。小区里有三个地方可以收快递:

  • 你家门口:只有你能拿,邻居拿不到
  • 小区快递柜:整个小区的业主都能用
  • 城市物流中心:全市都能用

变量作用域也是类似的概念。Python里的变量也分“放在哪”:

  • 局部变量:只属于当前函数,外面访问不到
  • 嵌套作用域:外层函数里的变量,内层函数可以访问
  • 全局变量:整个文件都能访问
  • 内置作用域:Python自带的,任何时候都能用

这就是著名的LEGB规则:Local → Enclosing → Global → Built-in。

别被这四个英文词吓到。用一个例子就能说清楚。


第一层:局部作用域(Local)

先看这段代码:

代码语言:javascript
复制
def greet():
    message = "你好"
    print(message)

greet()  # 输出:你好
print(message)  # 报错!NameError: name 'message' is not defined

message是在greet函数内部创建的,它只属于这个函数。函数执行完后,message就消失了。外面的代码看不到它。

这就是局部变量:函数内部定义的变量,外面访问不到。

反过来也一样:

代码语言:javascript
复制
name = "张三"

def say():
    age = 18
    print(name)  # 能读到外面的name
    print(age)   # 能读到自己的age

say()
print(age)  # 报错!外面读不到函数里的age

简单总结:函数里面能看到外面,外面看不到里面

就像一个房间:里面的人能看到窗外的街景,但街上的人看不到房间里放了什么。


第二层:全局作用域(Global)

全局变量定义在函数外面,整个文件里的代码都能访问它。

代码语言:javascript
复制
# 全局变量
app_name = "我的应用"
version = "1.0"

def show_info():
    print(app_name)  # 函数里可以读全局变量
    print(version)

show_info()  # 正常运行
print(app_name)  # 外面也能用

看起来很简单对吧?但坑马上就来了。

函数里可以读全局变量,但不能直接改

代码语言:javascript
复制
counter = 0

def increment():
    counter = counter + 1  # 报错!

increment()

为什么读可以,改不行?

因为当Python看到函数里有counter = xxx这种赋值语句时,它的第一反应是:“哦,你要在函数里创建一个新的局部变量,名字叫counter。”

然后它看到等号右边也有counter,就去局部作用域找这个新变量——还没创建完呢,怎么找得到?于是报错。

解决方法:明确告诉Python,“我要用的是外面的那个counter”。

代码语言:javascript
复制
counter = 0

def increment():
    global counter  # 声明:这个counter是全局的
    counter = counter + 1

increment()
print(counter)  # 1

global就像一把钥匙,让你走进全局变量的房间去修改它。


第三层:嵌套作用域(Enclosing)

函数里面还可以定义函数。这种嵌套函数里,就能访问外层函数的变量。

代码语言:javascript
复制
def outer():
    x = 10  # 外层函数的变量
    
    def inner():
        print(x)  # 内层函数可以读外层变量
    
    inner()

outer()  # 输出10

这就叫嵌套作用域(也叫闭包作用域)。inner函数可以访问outer函数里的x

但如果inner想修改x呢?

代码语言:javascript
复制
def outer():
    x = 10
    
    def inner():
        x = 20  # 这是在inner里创建了一个新的局部变量x
        print("inner里的x:", x)
    
    inner()
    print("outer里的x:", x)

outer()
# 输出:
# inner里的x: 20
# outer里的x: 10

和外层函数修改全局变量一样,默认情况下,内层函数对变量的赋值只会创建局部变量,不会影响外层函数的变量。

要修改外层函数的变量,需要用nonlocal

代码语言:javascript
复制
def outer():
    x = 10
    
    def inner():
        nonlocal x  # 声明:这个x来自外层函数
        x = 20
    
    inner()
    print(x)  # 20

outer()

nonlocalglobal很像,但nonlocal找的是外层函数的变量,不是全局变量。


第四层:内置作用域(Built-in)

这是Python自带的作用域,任何时候都能用。

代码语言:javascript
复制
print(len("hello"))  # len是内置函数
print(sum([1,2,3]))  # sum是内置函数
print(str(100))      # str是内置类型

你不需要导入任何东西就能用它们。

但有一个需要注意的地方:不要用内置函数的名字做变量名

代码语言:javascript
复制
len = 10  # 你定义了一个叫len的变量
print(len("hello"))  # 报错!因为len现在是10,不是函数了

这就叫“覆盖了内置作用域”。修复也很简单:换个变量名,或者删掉这个变量。


一张图看懂LEGB查找顺序

当你在代码里写一个变量名时,Python按这个顺序去找:

  1. Local:当前函数内部
  2. Enclosing:外层函数(如果有嵌套)
  3. Global:模块全局
  4. Built-in:Python内置

找到第一个匹配的就停下,找不到就报NameError

看个综合例子:

代码语言:javascript
复制
x = "global x"  # 全局

def outer():
    x = "outer x"  # 外层函数
    
    def inner():
        x = "inner x"  # 局部
        print(x)
    
    inner()

outer()  # 输出 "inner x"

如果注释掉inner里的x = "inner x"

代码语言:javascript
复制
def outer():
    x = "outer x"
    
    def inner():
        print(x)  # 没有局部x,就去外层找
    
    inner()

outer()  # 输出 "outer x"

再把outer里的x注释掉:

代码语言:javascript
复制
x = "global x"

def outer():
    def inner():
        print(x)  # 局部没有,外层没有,就去全局找
    
    inner()

outer()  # 输出 "global x"

最后,如果你定义了一个叫print的变量,Python也不会去找内置的print了:

代码语言:javascript
复制
print = "hello"
print("world")  # 报错,因为print现在是字符串,不能当函数调用

为什么要有作用域?直接全部用全局不好吗?

你可能觉得:作用域这么多层,好麻烦,为什么不把所有变量都设为全局?

答案很简单:混乱

想象一个5000行的文件,100个函数共享100个全局变量。你在修改一个函数时,根本不知道这个变量还在哪里被改过。改一处,崩十处。

作用域就是隔离区。每个函数有自己的小房间,房间里的变量只属于自己。这样你可以放心地写itempx这些名字,不用担心和别的函数冲突。

作用域也是一种文档。看到globalnonlocal,你就知道:“这个变量是共享的,改动它可能会影响其他地方。”


四个最容易踩的坑

坑1:以为if/for能创造作用域

代码语言:javascript
复制
if True:
    x = 10

print(x)  # 10,x还在!

for i in range(5):
    pass

print(i)  # 4,i还在!

很多语言里,iffor里的变量只在代码块内有效。Python不是。Python只有函数能创造新作用域,ifforwhilewith都不能。

坑2:在函数内意外创建局部变量

代码语言:javascript
复制
name = "张三"

def change():
    name = "李四"  # 这不是修改全局,是创建局部
    print("函数内:", name)

change()  # 函数内: 李四
print("全局:", name)  # 全局: 张三 —— 没变!

想改全局变量但忘记写global,Python不会报错,而是默默创建一个局部变量。这种“静默失败”比报错更难找。

坑3:在列表推导式里用变量

代码语言:javascript
复制
x = 10
squares = [x**2 for x in range(5)]
print(x)  # Python 3里还是10,Python 2里会变成4

Python 3修复了这个问题:列表推导式的变量不会泄漏到外部。但如果你用的是Python 2,就要小心。

坑4:在嵌套函数里用循环变量

代码语言:javascript
复制
funcs = []
for i in range(3):
    funcs.append(lambda: i)

for f in funcs:
    print(f())  # 2 2 2,不是0 1 2

所有lambda函数都记住了同一个i的最终值(2)。要解决这个问题,可以把i作为默认参数传进去:

代码语言:javascript
复制
funcs = []
for i in range(3):
    funcs.append(lambda i=i: i)  # 把当前i的值固定下来

for f in funcs:
    print(f())  # 0 1 2

实战:三个函数让你彻底明白

写三个简单的函数,涵盖了LEGB的所有情况。

函数1:只读

代码语言:javascript
复制
config = "production"

def get_env():
    return config  # 读全局,不需要global

print(get_env())  # production

函数2:修改

代码语言:javascript
复制
mode = "read"

def switch_to_write():
    global mode
    mode = "write"

switch_to_write()
print(mode)  # write

函数3:嵌套

代码语言:javascript
复制
def make_counter():
    count = 0
    
    def increment():
        nonlocal count
        count += 1
        return count
    
    return increment

counter = make_counter()
print(counter())  # 1
print(counter())  # 2
print(counter())  # 3

最后一个例子展示了一个重要模式:闭包increment函数“记住”了count变量,即使make_counter已经执行完了,count也没有消失。


快速判断一个变量的作用域

给你一个简单的判断方法:

  1. 这个变量在函数内有没有被赋值?
    • 没有 → 去外层/全局找
    • 有 → 它是局部变量(除非你用globalnonlocal声明了)
  2. 如果它是局部变量,赋值之前能不能用到它?
    • 不能 → 会报UnboundLocalError

记住这两条,90%的作用域问题都能解决。


一张表总结

场景

写法

关键字

查找范围

函数内读变量

直接写

局部→嵌套→全局→内置

函数内修改全局变量

global x; x = 10

global

全局

嵌套函数修改外层变量

nonlocal x; x = 10

nonlocal

外层函数

读取内置函数

直接用

内置命名空间


回到开头的那个bug。我朋友的问题出在哪?

代码语言:javascript
复制
total = 0

def process_numbers():
    # 这里少了一行
    while True:
        num = input("请输入数字:")
        if num == 'q':
            break
        total = total + int(num)  # total被赋值,被视为局部变量

total = total + 1这个语句让Python把total当作局部变量。但局部变量total在赋值前就被用到了,所以报错。

修复方法就是在函数开头加上:

代码语言:javascript
复制
def process_numbers():
    global total
    # 剩下的代码不变

加上这一行之后,total就不再是局部变量了,而是指向全局的那个total

我朋友后来把代码改好了,还学会了用globalnonlocal。从那以后,他再也没被作用域的问题困住过。

希望看完这篇文章的你,也不会。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一个让我调试了2小时的bug
  • 什么是作用域?一个快递站的比喻
  • 第一层:局部作用域(Local)
  • 第二层:全局作用域(Global)
  • 第三层:嵌套作用域(Enclosing)
  • 第四层:内置作用域(Built-in)
  • 一张图看懂LEGB查找顺序
  • 为什么要有作用域?直接全部用全局不好吗?
  • 四个最容易踩的坑
  • 实战:三个函数让你彻底明白
  • 快速判断一个变量的作用域
  • 一张表总结
相关产品与服务
云开发 CloudBase
AI-Native 的全栈应用开发平台,为 AI 驱动的开发流程和终端应用而设计。提供应用开发所需的完整 Serverless 资源,包括身份认证、数据库、云函数与容器、文件存储、Web 应用托管等能力;支持小程序/小游戏、AI Agent 应用、Web 应用等场景;提供 Skills/MCP 工具适配主流 AI 编程工具,避免了开发过程中繁琐的服务端搭建及运维,开发者可以专注于业务逻辑的实现,开发效率更高。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档