python中变量的查找顺序是legb规则。具体来说,1. local:函数内部定义的变量优先被使用,若未赋值前引用会报错;2. enclosing:嵌套函数外层作用域变量可访问,修改需用nonlocal;3. global:模块层级的全局变量,跨模块不可见;4. builtin:内置作用域,包含内置函数和变量,避免覆盖。合理使用作用域和声明关键字能有效避免错误。
在Python中,变量的查找顺序是按照LEGB规则来执行的。简单来说,就是当你在代码中使用一个变量时,Python会按照 Local → Enclosing → Global → Builtin 这个顺序去查找这个变量的值。
这听起来好像挺简单的,但实际写代码的时候,特别是函数嵌套、模块导入混在一起时,很容易出错。下面我们就从几个常见的场景出发,看看LEGB到底是怎么起作用的。
Local:局部作用域优先
Local指的就是函数内部定义的作用域。只要你在一个函数里面用 = 给变量赋值(除非特别说明是global或nonlocal),它就会被认为是局部变量。
举个例子:
立即学习“Python免费学习笔记(深入)”;
def func(): x = 10 print(x) func()
这里打印的是10,没问题。但如果你这样写:
def func(): print(x) x = 10 func()
这时候就会报错:UnboundLocalError: local variable ‘x’ referenced before assignment。
因为Python看到你在函数里给x赋值了,所以它认定x是一个局部变量。但在print的时候还没赋值,就出错了。
建议:
- 如果你想在函数里使用全局变量,记得加 global 声明。
- 如果你在嵌套函数里想修改外层函数的变量,要用 nonlocal。
Enclosing:嵌套函数的外层作用域
Enclosing指的是嵌套函数的外层函数的作用域。这是很多新手容易忽略的地方。
比如:
def outer(): x = "outer" def inner(): print(x) inner() outer()
这里inner函数能访问到outer里的x,这就是Enclosing作用域在起作用。
但如果我们在inner里试图修改x呢?
def outer(): x = "outer" def inner(): x = "inner" print(x) inner() print(x) outer()
结果会是:
inner outer
说明inner里的x是它自己的局部变量,并没有影响到outer里的x。
如果你希望inner修改outer中的x,就要加上 nonlocal:
def outer(): x = "outer" def inner(): nonlocal x x = "inner" print(x) inner() print(x) outer()
输出:
inner inner
Global:模块层级的全局变量
Global指的是在模块层级(也就是不在任何函数里)定义的变量。
比如:
x = "global" def func(): print(x) func()
这时打印的是 “global”,因为func找不到局部变量x,就去全局找。
但如果你在func里重新赋值x,又不加global,那又会触发前面说的那个错误。
常见误区:
- 模块A导入了模块B,模块B中的函数不能直接访问模块A的全局变量。
- 全局变量不是“整个程序”的变量,而是当前模块下的。
Builtin:内置作用域
Builtin是最底层的作用域,包含了Python自带的一些内置函数和变量,比如 print()、len()、True、Exception 等。
你可以把它理解为“最后兜底”的一层。
比如你写:
print(len("hello"))
这里的 len 就是从Builtin作用域里找到的。
但如果你不小心覆盖了内置变量,可能会出问题:
len = "oops" print(len("hello")) # 报错:TypeError: 'str' object is not callable
这时候你就把内置的 len 函数覆盖成了字符串,当然就不能用了。
建议:
- 不要随便用内置名称做变量名。
- 可以用 del len 来恢复,但这只是临时解决办法,最好一开始就避免。
基本上就这些。LEGB规则虽然看起来简单,但在函数嵌套、变量遮蔽等情况下容易让人掉坑。平时写代码时注意变量作用域,合理使用 global 和 nonlocal,就能避免很多莫名其妙的问题。
评论(已关闭)
评论已关闭