from: https://serholiu.com/python-closures html
闭包这个概念在 JavaScript 中讨论和使用得比较多,不过在 Python 中却不是那么显而易见,之因此说“不是那么”,是由于即便用到了,也没用注意到而已,好比定义一个 Decorator 时,就已经用到闭包了。网上对闭包的各类解释,感受很是晦涩,在这里谈谈个人浅显认识:要造成闭包,首先得有一个嵌套的函数,即函数中定义了另外一个函数,闭包则是一个集合,它包括了外部函数的局部变量,这些局部变量在外部函数返回后也继续存在,并能被内部函数引用。python
这是个常常使用到的例子,定义一个函数 generate_power_func
,它返回另外一个函数,如今闭包造成的条件已经达到。linux
1 2 3 4 5 6 |
def generate_power_func(n): print "id(n): %X" % id(n) def nth_power(x): return x**n print "id(nth_power): %X" % id(nth_power) return nth_power |
对于内部函数 nth_power
,它能引用到外部函数的局部变量 n
,并且即便 generate_power_func
已经返回。把这种现象就称为闭包。具体使用一下。shell
1 2 3 4 5 |
>>> raised_to_4 = generate_power_func(4)id(n): 246F770id(nth_power): 2C090C8>>> repr(raised_to_4)'<function nth_power at 0x2c090c8>' |
从结果能够看出,当 generate_power_func(4)
执行后, 建立和返回了 nth_power
这个函数对象,内存地址是 0x2C090C8,而且发现 raised_to_4
和它的内存地址相同,即 raised_to_4
只是这个函数对象的一个引用。先在全局命名空间中删除 generate_power_func
,再试试会出现什么结果。设计模式
1 2 3 |
>>> del generate_power_func>>> raised_to_4(2)16 |
啊哈,竟然没出现错误, nth_power
是怎么知道 n
的值是 4,并且如今 generate_power_func
甚至都不在这个命名空间了。对,这就是闭包的做用,外部函数的局部变量能够被内部函数引用,即便外部函数已经返回了。闭包
如今知道闭包是怎么一回事了,那就到看看闭包究竟是怎么回事的时候了。Python 中函数也是对象,因此函数也有不少属性,和闭包相关的就是 __closure__
属性。__closure__
属性定义的是一个包含 cell 对象的元组,其中元组中的每个 cell 对象用来保存做用域中变量的值。ide
1 2 3 4 5 6 |
>>> raised_to_4.__closure__(<cell at 0x2bf4ec0: int object at 0x246f770>,)>>> type(raised_to_4.__closure__[0])<type 'cell'>>>> raised_to_4.__closure__[0].cell_contents4 |
就如刚才所说,在 raised_to_4
的 __closure__
属性中有外部函数变量 n
的引用,经过内存地址能够发现,引用的都是同一个 n
。若是没用造成闭包,则 __closure__
属性为 None
。对于 Python 具体是如何实现闭包的,能够查看 Python闭包详解,它经过分析 Python 字节码来说述闭包的实现。函数
闭包特性有着很是多的做用,不过都是须要时才会不经意的用上,不要像使用设计模式同样去硬套这些法则。这篇文章按照本身的理解翻译至 Python Closures Explained,可能和原文有些不一样之处,若有疑惑,请查看原文。附上一些参考资料。spa
闭包的概念、形式与应用: 能够从其中了解闭包的应用翻译
Python闭包详解:从字节码出发了解 Python 闭包的实现机制
理解Javascript的闭包: 从 Javascript 的闭包中了解一些闭包特性,能够和 Python 做下对比