python3的local, global, nonlocal简析

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会声明.

  • 35
    点赞
  • 109
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值