闭包

2018-01-24  本文已影响0人  马梦里

闭包由两部分组成,一个是返回的函数,一个是原函数的内部变量(或参数)

闭包本身是一个晦涩难懂的概念,不过可以先简单的理解为:闭包就是一个定义在函数n内部的函数,闭包使得变量即使脱离了该函数的作用域范围也依然能被访问到。

>>> def ma_add(n):
...     return lambda x:x+n
... 
>>> add_3 = ma_add(3)
>>> add_3
<function ma_add.<locals>.<lambda> at 0x0210CD20>
>>> add_3(7)
10

这里的 lambda 函数就是一个闭包。在全局作用域中,add_3(7) 可以正常执行,并且返回值为 10,之所以返回 10 是因为在 ma_add 局部作用域中,变量 n 的值在闭包作用使得它在全局作用域也可以被访问到。
常规函数的闭包:

>>> def my_add(n):
...     def wrapper(x):
...         return x+n
...     return wrapper
...
>>> add_5 = my_add(5)
>>> add_5(2)
7

深入理解闭包:
嵌套函数:

def print_msg():
    # print_msg 是外围函数
    msg = "zen of python"

    def printer():
        # printer是嵌套函数
        print(msg)
    printer()
# 输出 zen of python
print_msg()

对于嵌套函数,他可以访问到其外层作用域中声明的非局部(non-local)变量,比如上面的 msg 变量可以被嵌套函数 printer 正常访问。但在函数之外如果 print(msg) ,就会报错。
闭包:
闭包使得变量在函数外依然可以被访问。也就是可以访问外层函数的变量。

def print_msg():
    # print_msg 是外围函数
    msg = "zen of python"
    def printer():
        # printer 是嵌套函数
        print(msg)
    return printer

another = print_msg()
# 输出 zen of python
another()

这段代码和上段代码输出一样。不同的是,内部函数被直接返回了,注意:返回的并不是结果,而是这个函数

一般情况下,函数中的局部变量仅在函数的执行期间可用。一旦 print_msg() 执行过后,msg 变量不可再用。然而,这里 print_msg 执行完后,在调用 another 的时候, msg 变量的值被正常输出了。
这段话的意思是,abother 现在相当于一个独立的函数,和 print_msg 函数是并列关系,但是 another 能够访问 print_msg 函数里面的变量 msg
another == printer

def printer():
        print(msg)
    return printer

这里的 another 就是一个闭包,闭包本质上是一个函数,它由两部分组成,printer 函数和变量 msg(print_msg 函数的参数也可以)。闭包使得这些变量的值始终保存在内存中。

闭包。顾名思义,就是一个封闭的包裹,里面包裹着自由变量,就像在类里面定义的属性值一样,自由变量的可见范围随同包裹,哪里可以访问到这个包裹,哪里就可以访问这个自由变量。

为什么要使用闭包

闭包避免了使用全局变量。此外,闭包允许函数与其所操纵的某些数据(环境)关联起来。这一点与面向对象编程非常相似,在面向对象编程中,对象允许我们将某些数据(对象的属性)与一个或者多个办法相关联。

一般来说,当对象中只有一个方法时,使用闭包是最好的选择:

def adder(x):
    def wrapper(y):
        return x + y
    return wrapper

adder5 = adder(5)
# 输出 15
adder5(10)
# 输出 11
adder5(6)

这比用类来实现更优雅,此外装饰器也是基于闭包的一种应用场景。

上一篇 下一篇

猜你喜欢

热点阅读