python global、nonlocal、闭包、作用域——第10天

今天工作的内容已经完成,于是在公司学习一下午,遇到的闭包作用域的问题一直想不明白,于是问了我们开发他也看了很久才明白(因为他是做java),然后心里想原来并不是所有人都是聪明的,别人 变成优秀的现在就是因为静下心去研究提升。所以自己想转行煎熬的内心又平静而坚定了。

 

一、变量闭包作用域

  python的闭包是如果在一个内部函数里,对外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包

def outer():
    x=8
    def inner(y):
        return x + y
    return inner

  下面这个应该是很多人都看过的,但是在我最初看的时候我以为是1,3,9,但是程序运行都是9,debug之后想了好久才明白

  

def count():
    fs = []
    for i in range(1,4):
        def fn():
            return i*i
        fs.append(fn)
    return fs
f1,f2,f3 = count()
print(f1()) #9
print(f2()) #9
print(f3()) #9

因为是fs存了fn这个函数,return i*i 这里的i使用了外部函数for循环里面的i,当调用fn函数的时候,for i 里面的i已经循环变成3, fn函数执行的时候去找变量i是i=3的,所以就都是9

想要把上边的结果变成1,4,9可以使用如下方法:

def count():
    def f(j):
        def g():
            return j*j
        return g
    fs = []
    for i in range(1, 4):
        fs.append(f(i)) # f(i)立刻被执行,因此i的当前值被传入f(),每一次的i都会指向一个g的内容地址
    return fs

f1,f2,f3 = count()
print(f1()) #1
print(f2()) #4
print(f3()) #922:40:53

下面的代码会报错:UnboundLocalError: local variable 'a' referenced before assignment

a = 10
def bar():
    print(a)
    a = 'in bar'
bar()

报错的原因是因为python编译的时候发现a局部变量是这样子定义的a = ,没有赋值。

在上面的函数bar中在执行第一条语句的时候,为什么不去访问全局变量a呢这是Python语法的规定,当在函数体中有赋值语句的时候,编译的时候就认为定义了局部变量,从而保证函数封装性

二、global

想要前面的值不报错可以使用global关键字

a = 10
def bar():
    global a
    a = 'in bar'
bar()
print(a) #in bar

但是使用这种代码得小心了,他很容易改变全局变量的值

三、nonlocal

闭包

def fn():
    count = 8
    def inner(dt=0):
        r =count +dt
        print(r) #8
    return inner
fn()()

如果上面的代码改动一点点他就会报上面的错:UnboundLocalError: local variable 'a' referenced before assignment,原理是和之前一样的

def fn():
    count = 8
    def inner(dt=0):
        count+=dt
        print(count) 
    return inner
fn()() #会报错

如果想要上面的代码不报错可以使用nonlocal关键字如下:

def fn():
    count = 8
    def inner(dt=0):
        nonlocal count
        count =count +dt
        print(count)
    return inner
fn()() #8

nonlocal关键字使用是,内部函数的变量没有时,可以到外部函数寻找变量。

但是值得注意的是使用nonlocal关键字时,内部函数的变量改变了外部函数的变量也会改变,如下:

name = 'global'
def test():
    name = 'local'

    def inner_test():
        nonlocal name
        # global name

        name = name + '变量'
        return name

    print(name) #'local'+'变量'
    return inner_test()

print(name) #'global'
print(test())
print(name) #'global'
def funX():
    x =5
    def funY():
        nonlocal x
        x+=1
        return x
    return funY
a = funX()
print(a()) #6
print(a()) #7 因为x+=1已经把外部函数的x改成了6
print(a()) #8

所以我感觉global和nonlocal还是少用的好

                       

 

 

posted @ 2019-08-02 23:39  智、心  阅读(489)  评论(0编辑  收藏  举报