闭包
所谓闭包,就是将组成函数的语句和这些语句的执行环境打包在一起时,得到的对象。
- 栗子1
def count():
fs = []
for i in range(1, 4):
def f():
return i*i
fs.append(f) #这里的 f 已经是一个函数 加入了
return fs
f1, f2, f3 = count()
>>> f1()
9
>>> f2()
9
>>> f3()
9
上面这个函数有点纠结!
纠结点1:f1, f2, f3 = count()是什么意思?
答:f1, f2, f3 = count() 这个并不是我猜想3个都平等同样的指向fs这个list,更有可能的是,这个三个这是分别指向了list中的3个元素
(1)只有元素才有可能是int
(2)如果只写成f1,f2 = count()
的话
会报错ValueError: too many values to unpack (expected 2)
写成f1,f2,f3,f4= count()
的话
会报错ValueError: need more than 3 values to unpack
(3)打印出来看
>>>print(f1,f2,f3)
返回3个<function count.<locals>.f at 0x00000000037F5400>
>>>f1= count
>>>print(f1())
[<function count.<locals>.f at 0x0000000002F65268>,
<function count.<locals>.f at 0x0000000002F65400>,
<function count.<locals>.f at 0x0000000002F65488>]
说明,如果返回一个列表,那么这个列表每个元素都是函数。
纠结点2:为什么f1(),f2(),f3()都返回9,即为啥列表中的元素都是9?
原因是返回函数引用了变量i,下面来解析下f1,f2,f3=count()
这句的执行过程:
当i=1, 执行for循环, 结果返回函数f的函数地址,存在列表fs中的第一个位置上。
当i=2, 由于fs列表中第一个元素所指的函数中的i是count函数的局部变量,i也指向了2;然后执行for循环, 结果返回函数f的函数地址,存在列表fs中的第二个位置上。
当i=3, 同理,在fs列表第一个和第二个元素所指的函数中的i变量指向了3; 然后执行for循环, 结果返回函数f的函数地址,存在列表fs中的第三个位置上。
所以在调用f1()的时候,函数中的i是指向3的:
f1():
return 3*3
同理f2(), f3()结果都为9
注意:返回闭包时牢记的一点就是:返回函数不要引用任何循环变量,或者后续会发生变化的变量。
- 栗子2:如果一定要引用循环变量怎么办?方法是再创建一个函数,用该函数的参数绑定循环变量当前的值,无论该循环变量后续如何更改,已绑定到函数参数的值不变:
def count():
def f(j):
def g():
return j*j
return g
fs = []
for i in range(1, 4):
fs.append(f(i)) # f(i)立刻被执行,因此i的当前值被传入f()
return fs
注意:当i=1时,f(1)即让j指向1,f(1)立即被执行,返回值是g函数,g函数作为参数传入fs.append()里
当i=2时,f(2)即让j指向2,此时j不是count的局部变量,不会影响到i=1是f(1)中j的指向。即函数f的参数绑定循环变量当前的值, 而不是循环变量本身。
>>> f1, f2, f3 = count()
>>> f1()
1
>>> f2()
4
>>> f3()
9