闭包简单来说就是内层函数引用了外层函数的参数,并最终返回内层函数这样的一种结构。
这个结构有一个特点,那就是函数不会立刻执行,要理解这个意思,其实就是理解 f 和 f() 的区别,f 不会执行函数而 f() 会执行函数,计算结果。
廖雪峰的教程中例子所要说明的问题就是:返回的函数的时候,相关参数和变量都保存在返回的函数,并且返回的函数并不会立刻执行,而是直到调用了f()
才执行。
def count():
fs = []
for i in range(1, 4):
def f():
return i*i
fs.append(f)
return fs
f1, f2, f3 = count()
你可能认为调用 f1(),f2() 和 f3() 结果应该是1,4,9但实际结果都是9。
分析原因:首先要注意到 fs 中 append 的是函数 f 而不是 f() 得到的值,而函数 f 实际上是 i 的平方,于是每次向 fs 中添加的都是计算 i 的平方的函数 f,由于函数 f 并不会立刻执行,只有 f() 才会执行,因此等到 for 循环结束后,i = 3,最终执行 fs 里面的函数 f1,f2 以及 f3 的时候,得到的结果都是 9。
因此注意:返回的函数是惰性的,不会立刻执行!
如果要引用循环变量的值怎么办,那就是让 f 执行,具体就是:
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
fs 添加的是 f(i),i 是循环变量,是不同的,而且 f(i) 是执行的 ,因此,fs 中保存的 j*j 是已经计算出来了的,j 的值分别是循环变量的值,也就是 fs 中存的就是 [1*1,2*2,3*3],但由于 g 是函数,还没有计算,因此到执行 f1(),f2(), f3() 的时候分别输出 1,4,9