python3中的nonlocal 与 global

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

posted @ 2020-04-15 22:23  friedCoder  阅读(986)  评论(0编辑  收藏  举报