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中修改外部变量挺麻烦的,因此最好使用面向对象的编程方法,而非面向过程。