引子
大概一个月前,一个同事问,什么是"闭包",然后整个月好像都在纠结这个问题啊喂。。。其实搜搜概念,网上有一大堆,研究一下其实也不复杂。然后不知道什么鬼总感觉这东西跟装饰器有什么数不清楚的关系啊(后面写一篇),于是,又折腾了一下,然后,尼玛的另一个同事问,"为什么我们需要闭包"
然后我就无言以对啊,搞明白一个概念,如果没有什么实践价值,有个卵用啊于是在接下来的时间不停的想啊想啊想。。。终于今晚想到了一个绝佳的例子,怎么样也要记录一下吧
例子
废话不多suo了。直接上例子吧。比如math包里面有个log函数,可是我并不知道log2(8)这玩意要写成math.log(2, 8)
还是math.log(8, 2)
,而且我也不想log2写一个log10再写一个吧。。。所以,为了少码几个字,我们可以这样写:
import math
def my_log(a):
def wrapper(N):
return math.log(N, a)
return wrapper
if __name__ == "__main__":
log2 = my_log(2)
lg = my_log(10)
print log2(8)
print lg(100)
是的,通过log2或者lg,就可以直接计算以2为底或者以10为底的数的对数了。而且log2(8)
这种写法,也比较符合正常人的想法。顺带,我们也不需要再单独定义两个函数了,O(∩_∩)O~
其实,估计不用闭包也会有办法实现同样的功能,但是闭包绝对提供了一种更为优(tou)雅(lan)的实现方式。下面我们说说什么是闭包。
定义
反正,我是不会告诉你,它的官方定义是:
如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure)
如果要用一个例子来解释上面这段话的话,可以参考以下代码:
def outer(outer_para):
def inner(inner_para):
return outer_para + inner_para
return inner
if __name__ == "__main__":
func = outer(5)
print func(10)
接下来再补充完善一下:
- 必须有一个内嵌函数(nested function),也就是在函数内部定义一个函数,比如例子中的inner
- 必须在内嵌函数中调用上层函数中的变量,比如在inner中调用outer_para
- 上层函数必须返回内嵌函数,也就是outer的最后一行必须是
return inner