python3的local, global, nonlocal简析

python 专栏收录该内容
72 篇文章 1 订阅

python3:变量作用域及global,nonlocal的用法

在Python程序中声明、改变、查找变量名时,都是在一个保存变量名的命名空间中进行中,此命名空间亦称为变量的作用域。python的作用域是静态的,在代码中变量名被赋值的位置决定了该变量能被访问的范围。即Python变量的作用域由变量所在源代码中的位置决定.
变量作用域之LENGB

L = Local     局部作用域
E = Enclosing 嵌套作用域
N = nonlocal  只作用于嵌套作用域,而且只是作用在函数里面
G = global    全局作用域
B = Built-in  内置作用域

python引用变量的顺序: 当前作用域局部变量->外层作用域变量->当前模块中的全局变量->python内置变量.

下面讲分别举例说明:
1. L = Local     局部作用域
局部变量包含在def定义的函数体内。在函数体内声明的变量,默认都是局部变量,除非有特别说明,如全局变量的声明要用关键字global.

def testlocal():
    x = 123 # x即为局部变量
print(x)

以上代码运行就会报NameError, 因为x是个局部变量,在函数外要print(x), 按照上面python引用变量的顺序,是找不了x变量的,故为报错. 

NameError: name 'x' is not defined

1.1 再举一例:

x = 123
def testlocal():
    print(x)

testlocal()

以上代码运行结果为123, 因为按照上面python引用变量的顺序,是找不到x变量的,故会向外层作用域找,就会找到x=123

1.2 如果将代码改为:

x = 123
def testlocal():
    print(x)
    x = 100

testlocal()

这时以上代码运行就会报错. 

UnboundLocalError: local variable 'x' referenced before assignment

什么原因?新手常会遇到这类问题, 因为按照python引用变量的顺序,当print(x)时,就会先在函数体内查找x,能找到,但在他的下一行,故为在声明前被引用是不对的.  
 

2. G = global    全局作用域
全局作用域的变量怎么用呢?
先看实例:

x = 123
def testGlobal():
    print(x)
    x = 100
    
testGlobal()

这就是上面1.2的实例,会报错,如果将其为:

x = 123
def testGlobal():
    global x
    print(x)
    x = 100
    
testGlobal()

试试看,结果运行结果为123, 是不会报错的,自已体会一下吧. 

3. nonlocal的用法
这个一般是用在闭包函数里. 但是一定要认识到, nonlocal声明的变量在上级局部作用域内,而不是全局定义, 如果在它声明的变量在上级局部中不存在,则会报错. 实例如下:

x = 123
def outer():
    x = 100
    def inter():
        x = 200
    inter()
    print(x)
    
func()

运行的结果为100, 那如果要求inter中对x的修改是有效的,必须加上关键字nonlocal

x = 123
def outer():
    x = 100
    def inter():
        nonlocal x
        x = 200
    inter()
    print(x)
    
func()

运行的结果为200, 符合预期.

现将inter的上级局部变量x去掉,则会报错

x = 123
def outer():
    # x = 100
    def inter():
        nonlocal x
        x = 200
    inter()
    print(x)
    
func()

4. 其它.
python变量的作用域只在作用域内有效,需要注意的是,在if-elif-else, for-in, while, try-except-finally这些关键字中并不会产生特定的作用域,如下:

#!/user/bin/python3
#-*- coding:utf-8 -*-

for i in range(10):
    i = i + 1
print(i)
>>>10

运行结果并不会报错,会输入i, 这里的i就是一个全局变量,这是python的一个特点,如果在C/java中这种写法就会报错,会报变量i会声明.

评论2
请先登录 后发表评论~
©️2021 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页

打赏作者

五力

你的鼓励将是我创作的最大动力

¥2 ¥4 ¥6 ¥10 ¥20
输入1-500的整数
余额支付 (余额:-- )
扫码支付
扫码支付:¥2
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值