闭包是指延伸了作用域的函数,其中包含函数定义体中引用、但是不在定义体中定义的非全局变量。函数是不是匿名没有关系,关键是它能访问定义体之外定义的非全局变量。
举个例子:计算移动平均值的类
def make_averager():
series = []
def averager(new_value):
series.append(new_value)
total = sum(series)
return total/len(series)
return averager
>>> avg = make_averager()
>>> avg(10)
10.0
>>> avg(11)
10.5
>>> avg(12)
11.0
>>> avg.__code__.co_varnames
('new_value', 'total')
>>> avg.__code__.co_freevars
('series',)
>>> avg.__closure__ # doctest: +ELLIPSIS
(<cell at 0x...: list object at 0x...>,)
>>> avg.__closure__[0].cell_contents
[10, 11, 12]
结果如上,avg 如何保存10,11,12,这个函数是可以通过类来实现的,但是上面使用的是高阶函数 make_averager()实现。
调用 make_averager 时,返回一个 averager 函数对象。每次调用 average 时,他会把参数添加到系列值中,计算当前平均值。series 是 make_averager 函数的局部变量。在 averager 函数中,series 是自由变量,这一点很重要,列表本身是可变的,如果是 str 等不可变量类型,要加上 nonlocal,否则是无法出现在闭包中的。
averager 闭包延伸到函数的作用域之外,包含自由变量 series 的绑定。它实现了函数之中的函数调用内函数之外的变量,这一点配合装饰器有很多的用处。