理解Python中的闭包

2017-11-06  本文已影响43人  DamonYJ

Python基础

Python中,函数也是一个对象,而且函数对象可以被赋值给变量,所以,通过变量也能调用该函数。
例如:

def now():
    print('2017-11-06')

f = now
f()    # 打印 2017-11-06

理解什么是闭包?

Python中,闭包的概念是:
1.首先是个嵌套函数
2.内层函数使用了外层函数的变量或者参数
3.外层函数把内层函数当做返回值进行返回

一段简单的闭包代码示例

def func_outer(): # 外部函数
    
    temp = 666 # 外部函数的变量

    def func_inner(): # 嵌套的内部函数
        print(temp) # 使用了外部函数的变量

    return func_inner # 返回内部函数
    

test = func_outer()
test() # 打印666

由以上可以看出,test变量其实就是返回的func_innter函数对象,test()就是通过这个变量来调用func_innter函数.

闭包的注意点:

请思考一下闭包中下面的几种情况

1.在内层函数修改了外层函数的变量
2.在闭包中,引用了一个可能会发生变化的变量

先看第一种情况,对代码进行修改一下

def func_outer():
    temp = 666
    def func_inner():
        temp = 888 # 0.在内层函数中修改了外层函数的变量的值

    print(temp)  # 1
    func_inner() #2. 在外部就执行了内部函数
    print(temp) # 3.

    return func_inner


func_outer()

在注释3处,打印的值还是666,这个问题比较简单,因为在注释0代码处相当于重新声明了一个局部变量temp,所以这只是一个作用域的问题(如果你是编程新手的话请打开你的搜索引擎了解作用域的知识)
其实很简单,就四个大字: 就近原则

如果要修改外部函数的temp变量使用关键字nonlocal声明:
注释0处代码改为

nonlocal temp
temp = 888

第二种情况,先看代码:

def func_outer():
    temp = 666
    def func_inner():
        print(temp)

    # 修改了变量的值
    temp = 888    
    return func_inner
    
test = func_outer()
test()

上面代码在声明内部函数func_inner()之后修改了temp变量的值,然而打印结果却是888.

要想搞懂原因的话可以先写一个小测试

def test_func():
    print(nondefine)

上面函数打印了一个未被声明过的变量,但是,如果就这样运行的话程序时没有问题的,照样能正常运行.而当我们调用这个函数的时候test_func(),再次运行程序就报错了.报错原因

NameError: name 'nondefine' is not defined

上面的测试证明了当一个函数被调用时,它内部变量对应的值才会被确定,这就能解释上面为什么会打印888了.

上一篇下一篇

猜你喜欢

热点阅读