闭包的实际应用
闭包
闭包闭包顾名思义封闭的包裹,那它到底是个啥包裹呢
首先了解闭包前 我们先了解几个概念
作用域
作用域是程序运行时变量可被访问时的范围,定义在函数内部的是局部变量,局部变量的作用范围只能是函数内部范围内,它不能
在函数外部引用。
定义在模块最外层的变量是全局变量,它是全局范围内可见的,当然在函数里面也可以读取到全局变量的。例如:
num = 10 #全局变量
def foo():
print(num)
而在函数外部则不可以访问局部变量。例如:
def foo():
num = 10
print(num)
NameError: name 'num' is not defined
嵌套函数
函数不仅可以定义在模块的最外层,还可以定义在另外一个函数的内部,像这种定义在函数里面的函数称之为嵌套函数(nested function)例如:
def print_msg():
#print_msg 是外围函数
msg = 'liu of python'
def printer():
#printer是内嵌函数
print(msg)
printer()
print_msg()
输出为:
liu of python
可以看出外围函数的局部变量是嵌套函数的全局变量,所以内部函数里是可以访问到外围函数的局部变量的
那有没有一种即使不依附在外围函数下 也能访问到它的局部变量呢,答案就是闭包!
什么是闭包
我们来看一个例子:
def print_msg():
#print_msg 是外围函数
msg = 'liu of python'
def printer():
#printer是内嵌函数
print(msg)
return printer
another = print_msg()
another()
输出为:
liu of python
一般情况下,函数中的局部变量仅在函数的执行期间可用,一旦 print_msg() 执行过后,我们会认为 msg变量将不再可用。然而,在这里我们发现 print_msg 执行完之后,在调用 another 的时候 msg 变量的值正常输出了,这就是闭包的作用,闭包使得局部变量在函数外被访问成为可能。
看完这个例子,我们再来定义闭包,维基百科上的解释是:
在计算机科学中,闭包(Closure)是词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。
为什么要使用闭包
闭包避免了使用全局变量,此外,闭包允许将函数与其所操作的某些数据(环境)关连起来。这一点与面向对象编程是非常类似的,在面对象编程中,对象允许我们将某些数据(对象的属性)与一个或者多个方法相关联。
一般来说,当对象中只有一个方法时,这时使用闭包是更好的选择。来看一个例子:
def adder(x):
def wrapper(y):
return x + y
return wrapper
adder5 = adder(5)
# 输出 15
print(adder5(10))
# 输出 11
print(adder5(6))
所有函数都有一个 closure属性,如果这个函数是一个闭包的话,那么它返回的是一个由 cell 对象 组成的元组对象。cell 对象的cell_contents 属性就是闭包中的自由变量。
>>> adder.__closure__
>>> adder5.__closure__
(<cell at 0x103075910: int object at 0x7fd251604518>,)
>>> adder5.__closure__[0].cell_contents
5
这解释了为什么局部变量脱离函数之后,还可以在函数之外被访问的原因的,因为它存储在了闭包的 cell_contents中了。