python3中的nonlocal 与 global

时间:2024-04-30 07:43:23

nonlocal 与 global

nonlocal翻译是非本地,global翻译是全局,它们都是python3的新特性。如果以类C语言的思维去看这2个关键字,很可能觉得它们差不多。但实际上它们很不一样。

下面先说说global。顾名思义,global能将一个变量提升为全局, 但是这个关键字有个特别的语法要求,即不能同时定义和声明,如global a=1。

a = 1
def func():
print(a)
global a
a += 2 # SyntaxWarning: name 'a' is used prior to global declaration
func() #输出1
print(a) #输出3

不熟悉python的可能会举得上面的global有点多余,可能会倾向于下面这种写法:

a = 1
def func():
a += 2 func()
print(a)

可惜这么写只会报错UnboundLocalError: local variable 'a' referenced before assignment, 具体原因可以参考https://www.cnblogs.com/friedCoder/p/12571983.html

这种类C写法不能照搬到Python中,毕竟python是解释型语言,两者不可完全类比。

对于global再补充最后一个小点

def func():
global a
a = 1 print(a) #NameError: global name 'a' is not defined def func():
global a
a = 1 func()
print(a) #输出1

可以发现python竟然还能直接在函数里面定义外部的全局变量,当然没有global是做不到的,不过这种写法很不好。


下面再来看看nonlocal

既然已经有一个处理全局的关键字了,那nonlocal就是一个处理局部变量的关键字。

a = 1
def outer():
a = 2
def inner():
print(a)
nonlocal a
a = 3
inner()
print(a) outer()
print(a)

发现执行报错SyntaxError: name 'a' is used prior to nonlocal declaration, 因为函数inner里面a在声明nonlocal之前就被print调用了,这是不允许的, global一样。

a = 1
def outer():
a = 2
def inner():
nonlocal a
a = 3
inner()
print(a) #3 outer()
print(a) #1

可见nonlocal就是能引用外部环境的局部变量,遵守最近原则。

注意这里强调局部变量,下面都是不允许的:

a = 1
def outer():
def inner():
nonlocal a
a = 3
inner()
print(a) outer() #################################### def outer():
global a
def inner():
nonlocal a
a = 3
inner()
print(a) outer()

报错SyntaxError: no binding for nonlocal 'a' found

从上面可以看出python中修改外部变量挺麻烦的,因此最好使用面向对象的编程方法,而非面向过程。