Python作用域

博客介绍了Python变量作用域的决定因素,指出def、class、lambda能直接改变作用域,if、while、for则不能。重点讨论了LEGB准则,说明函数内使用未认证变量时按此准则搜索。还提及global、nonlocal语句,如global语句可让def内的变量引用并修改全局变量值。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

明确一点,一个变量的作用域总是在被赋值的地方所决定。能直接改变变量作用域的有def、class、lambda,if、while、for不能直接改变变量作用域。

一、LEGB准则(LNGB准则)

这里重点讨论def语句,我们定义的变量实际上只有三种不同的作用域:

  1. 在def内部赋值,它被定为在函数内部,为本地Local
  2. 在一个嵌套的def中赋值,对于嵌套函数来说,它是非本地的Nonlocal
  3. 在def外部赋值,它就是整个文件全局的Global

L:Local、E:Enclose、G:Global、B:Built-in

根据LEGB准则,在函数内使用未认证的变量时,Python按照LEGB准则,依次搜索,直到第一次能找到这个变量的地方停下。

x = 100


def foo():
    x = 99
    print(x)


if __name__ == '__main__':
    foo()

结果为

99

 foo()内创建局部变量x跟全局变量x同名,foo()优先搜索局部变量x,输出99

x = 100


def foo():
    x = 99

    def bar():
        x = 98
        print(x)
    bar()


if __name__ == '__main__':
    foo()

结果

98

 调用foo(),返回bar(),bar()内部创建local变量x=98,对于bar()而言x=99是enclose,x=100是global的

优先输出local的x值98

如果删掉x=98,则会优先搜索E

x = 100


def foo():
    x = 99

    def bar():
        print(x)
    bar()


if __name__ == '__main__':
    foo()

结果是

99

非常类似C语言的局部作用域屏蔽全局作用域

二、global、nonlocal语句

x = 100


def foo():
    global x
    x = 99


if __name__ == '__main__':
    foo()
    print(x)

结果是

99

global语句允许def内部的x直接引用全局变量 x,并且修改了对象的值

 

 

 

 

 

 

<think>好的,用户想了解Python中的作用域规则LEGB以及使用方法。让我先回忆一下相关知识点。LEGB代表Local、Enclosing、Global、Built-in四个作用域层级。用户之前询问过列表的常用方法,现在转向作用域,可能需要基础到应用的全面解释。 首先,我需要确保正确理解LEGB的顺序。根据引用的资料,查找变量时先检查局部作用域,然后是嵌套函数的作用域(如果有的话),接着是全局,最后是内置作用域。比如在函数内部定义的变量属于Local,如果没找到,就会去外层函数找Enclosing,再找模块级的Global,最后是Python内置的Built-in。 用户可能需要具体的例子来理解每个层级。比如,在函数里引用一个变量,Python如何逐层查找。例如,在函数inner里使用变量,如果在local没有定义,就会查看outer函数的作用域,然后是全局,最后是内置函数如len或print。 同时,用户可能对全局变量和局部变量的区别有疑问,如何修改全局变量需要使用global关键字,或者nonlocal在闭包中的使用。这些是常见的易错点,需要详细说明。 另外,用户可能想知道实际应用中如何正确使用作用域规则,比如避免变量覆盖,或者合理设计函数嵌套。例如,在闭包中使用nonlocal来修改外部函数的变量。 还要考虑用户可能的误区,比如在函数内部意外覆盖全局变量,或者不理解为什么在函数内部修改全局变量需要声明global。这时候需要用例子说明声明前后的区别。 另外,根据引用内容,LEGB的实例分析很重要。比如,引用5中的例子,函数内部找不到变量就去全局找。可以构造类似的例子,帮助用户理解查找顺序。 最后,用户可能需要知道常见错误及解决方法,比如UnboundLocalError,以及如何通过变量声明来避免。同时,比较global和nonlocal的区别,以及它们的适用场景。 总结下来,回答的结构应该先解释LEGB每个层级,用代码示例说明,然后讨论变量声明(global和nonlocal),最后给出实践建议和常见问题。确保引用相关的资料,如引用1、2、3等,来支持解释的准确性。同时,注意用户之前的问题是关于列表方法,现在转向作用域,可能需要更基础的讲解,避免过于深入高级话题,但也要覆盖必要的细节。</think>### Python作用域规则LEGB详解 Python作用域规则遵循**LEGB原则**,这是变量查找的核心机制。通过以下分层解析和实例演示,帮助您掌握作用域的实际应用方法。 --- #### 一、LEGB层级定义 1. **L-Local(局部作用域)** 定义在**函数内部**的变量,仅在当前函数内有效 ```python def func(): x = 10 # 局部变量 print(x) # 输出10 func() # print(x) # 报错NameError ``` 2. **E-Enclosing(嵌套作用域)** 外层嵌套函数的变量(闭包环境) ```python def outer(): e_var = "闭包变量" # E层 def inner(): print(e_var) # 访问外层函数变量 inner() outer() # 输出"闭包变量" [^4] ``` 3. **G-Global(全局作用域)** 定义在**模块层级**的变量,跨函数共享 ```python g_var = "全局变量" # G层 def show_global(): print(g_var) # 直接访问全局变量 show_global() # 输出"全局变量" [^4] ``` 4. **B-Built-in(内置作用域)** Python预定义的函数/变量(如`print`, `len`) ```python def calc_length(s): return len(s) # 调用内置函数len [^3] print(calc_length("test")) # 输出4 ``` --- #### 二、变量查找顺序 Python按**L→E→G→B**顺序查找变量,找到即停止: ```python x = "全局变量" # G层 def outer(): x = "闭包变量" # E层 def inner(): x = "局部变量" # L层 print(x) # 优先输出局部变量 inner() outer() # 输出"局部变量" ``` --- #### 三、关键操作规范 1. **访问全局变量** 使用`global`声明修改全局变量: ```python count = 0 def increment(): global count # 声明使用全局变量 count += 1 increment() print(count) # 输出1 [^4] ``` 2. **修改嵌套作用域变量** 使用`nonlocal`声明修改闭包变量: ```python def outer(): total = 0 # E层 def inner(): nonlocal total # 声明修改闭包变量 total += 5 inner() print(total) # 输出5 outer() ``` --- #### 四、典型场景与避坑指南 1. **变量覆盖问题** 局部变量与全局变量同名时,默认访问局部变量: ```python value = 100 def test(): value = 200 # 创建新局部变量 print(value) # 输出200 test() print(value) # 仍输出100 [^2] ``` 2. **UnboundLocalError** 未声明`global`时尝试修改全局变量会报错: ```python total = 10 def update(): # 尝试修改全局变量会报错 total = total + 1 # UnboundLocalError ``` --- #### 五、作用域对比表 | 作用域类型 | 定义位置 | 生命周期 | 访问范围 | 修改方式 | |------------|--------------------|-------------------|------------------|----------------| | Local | 函数内部 | 函数执行期间 | 仅限当前函数 | 直接修改 | | Enclosing | 外层嵌套函数 | 外层函数执行期间 | 嵌套函数内部 | `nonlocal`声明 | | Global | 模块层级 | 程序运行期间 | 跨模块访问 | `global`声明 | | Built-in | Python解释器预定义 | 始终存在 | 全局可用 | 不可修改 | --- #### 六、最佳实践建议 1. **避免过度使用全局变量** 优先使用函数参数和返回值传递数据 2. **闭包中合理使用nonlocal** 明确修改外层作用域变量时需声明 3. **命名规范** 全局变量使用全大写命名(如`CONFIG`)提高可读性 ```python # 推荐写法示例 def calculate(data): result = sum(data) # 使用局部变量 return result * 2 GLOBAL_FACTOR = 1.5 # 明确标识全局变量 [^4] ``` --- ### 相关问题 1. 如何在类方法中访问全局变量? 2. `global`和`nonlocal`关键字的区别是什么? 3. 什么是Python的闭包陷阱?如何避免?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值