:>>>当只有执行函数的时候才会去找当时(执行)的值
: >>>核心把握:function对象存储两部分内容:一部分是环境变量__closure__,一部分是函数代码本身
:>>>闭包函数运行的时候使用的参数是编译的时候存储到内存中的函数环境变量
首先我们看下闭包的解释:
闭包是由函数和与其相关的引用环境组合成的对象
#!/usr/bin/env python2.7
# -*- coding=utf-8 -*-
# 例子转自:https://www.jianshu.com/p/41e3ab1fe956
def f1():
name1 = 'Alice'
name2 = 'Bob'
def f2():
print('hello,%s,%s'%(name1, name2))
return f2
if __name__ == '__main__':
name1 = 'David'
name2 = 'Frank'
func = f1()
func()
print(func.__closure__)
print(func.__closure__[0].cell_contents)
print(func.__closure__[1].cell_contents)
>>>
hello,Alice,Bob
(<cell at 0x03A55CF0: str object at 0x03CBEA20>, <cell at 0x03CBEAD0: str object at 0x03CBEAA0>)
Alice
Bob
"""
例子中的f2函数定义在f1函数内部,并且调用了f1内定义的变量name1和name2,f1返回了f2这个函数对象。
根据上边的代码__closure__可以看出返回的时候不仅将对应的函数对象返回也将当时引用的f1的变量返回了。这个就是其他博主提到的可以保存当时的环境变量(即使f1已经执行完毕被释放掉了)
从主函数重新定义name1和name2也可以看出,return f2的时候也顺带着把当时f1的name1和name2捆绑打包在一起了
"""
不同的引用环境和相同的函数组合可以产生不同的对象
# 此例子转自伯乐在线
def line_def(a, b):
def line(x):
return a * x + b
return line
if __name__ == '__main__':
func1 = line_def(3, 3) # line的实例1 (和a, b变量组成不同的实例)
func2 = line_def(2, 3) # line的实例2
print(func1(5))
print(func2(5))
# 输出
18
13
"""
这个也就是上边那句话的代码解释
"""
最后再次敲黑板,划重点:
闭包的最大特点是可以将外部函数的变量与内部函数绑定,并返回绑定变量后的函数(也即闭包),此时即便生成闭包的环境(外部函数)已经释放,闭包仍然存在。
闭包最大的应用就是装饰器,虽然还没写到装饰器的部分,但是这个先了解一下不吃亏!