1、预备知识——作用域(global vs local)
作用域是程序运行时变量可被访问的范围,定义在函数内的变量是局部变量。
global:
num = 10 # 全局作用域变量
def foo():
print(num)
foo() # 输出10
local:
def foo():
num = 10
print(num) # 会报错 NameError: name 'num' is not defined
2、预备知识——嵌套函数
定义在函数里面的函数称之为嵌套函数(nested function)。例如:
def print_msg():
# print_msg 是外围函数
msg = "zen of python"
def printer():
# printer是嵌套函数
print(msg)
printer()
print_msg() # 输出 zen of python
3、知识点考点——闭包
函数可以作为函数的返回值返回的函数称为闭包。例如:
def print_msg():
# print_msg 是外围函数
msg = "zen of python"
def printer():
# printer 是嵌套函数
print(msg)
return printer
another = print_msg()
another() # 输出 zen of python
一般情况下,函数中的局部变量仅在函数的执行期间可用,一旦 print_msg() 执行过后,我们会认为 msg变量将不再可用。然而,在这里我们发现 print_msg 执行完之后,在调用 another 的时候 msg 变量的值正常输出了,这就是闭包的作用,闭包使得局部变量在函数外被访问成为可能。
维基百科上的解释是:在计算机科学中,闭包(Closure)是词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。
这里的 another 就是一个闭包。
闭包,就是一个封闭的包裹,里面包裹着自由变量,就像在类里面定义的属性值一样,自由变量的可见范围随同包裹,哪里可以访问到这个包裹,哪里就可以访问到这个自由变量。
4、使用闭包的好处
闭包避免了使用全局变量,此外,闭包允许将函数与其所操作的某些数据(环境)关连起来。这一点与面向对象编程是非常类似的,在面对象编程中,对象允许我们将某些数据(对象的属性)与一个或者多个方法相关联。
一般来说,当对象中只有一个方法时,这时使用闭包是更好的选择。例如:
def adder(x):
def wrapper(y):
return x + y
return wrapper
if __name__ == '__main__':
adder5 = adder(5)
print(adder5)
# 输出 15
print(adder5(10))
# 输出 11
print(adder5(6))