python 闭包问题

问题:下面这段代码的输出结果将是什么?

def multipliers():
	return [lambda x: i * x for i in range(4)]

print([m(2) for m in multipliers()])

答:上面代码输出的结果是 [6, 6, 6, 6](不是我们想的 [0, 2, 4, 6])。
上述问题产生的原因是 Python 闭包的延迟绑定。这意味着内部函数被调用时,参数的值在闭包内进行查找。因此,当任何由 multipliers()返回的函数被调用时,i 的值将在附近的范围进行查找。那时,不管返回的函数是否被调用,for 循环已经完成,i 被赋予了最终的值 3。因此,每次返回的函数乘以传递过来的值 3,因为上段代码传过来的值是 2,它们最终返回的都是 6 即 (3*2)。

问题:什么是闭包?

在函数内部再定义一个函数,并且这个函数用到了外边函数的变量,那么将这个函数以及用到的一些变量称之为 闭包。

问题:如何修改上面的 multipliers 的定义从而产生想要的结果?

  • 一种解决方法就是用 Python 生成器。
def multipliers_1():
	for i in range(4):
		yield lambda x: i * x

print([m(2) for m in multipliers_1()])
  • 另外一个解决方案就是创造一个闭包,利用默认参数立即绑定。
def multipliers_2():
	return [lambda x, i=i: i * x for i in range(4)]

print([m(2) for m in multipliers_2()])