li = []
for i in range(0,5):
li.append(lambda : i**2)
print(li[2]())
print(li[4]())
说明:这段代码按照一般理解,应该得到一个包含了i**2的列表
- 先说li[]
[<function <lambda> at 0x7fa6b41c5ea0>, <function <lambda> at 0x7fa6b46c0840>, <function <lambda> at 0x7fa6b46c0620>, <function <lambda> at 0x7fa6b46c07b8>, <function <lambda> at 0x7fa6b46c08c8>]
这时我们对li[2]和li[4]进行求值,会得到两个同样的值16,如果按照普通的理解应该是两个4,16
这是因为lambda表达式写的是 lambda : i**2,此时的lambda中的i不是内部定义的变量,而是调用的时候引入定义在外部作用域的变量i,并且i是lambda调用时候访问的也就是说并不是调用时候定义访问的。而循环结束的时候i是4,所以无论怎么求值也都是最后4的平方结果。
所以,接下来我们做个改进
li1 = []
for i in range(0,5):
li1.append(lambda n=i: n**2)
print(li1[2]())
print(li1[4]())
这时我们可以见得lambda的表达式中相当于将定义于外部作用域的i变量暂时保存于局部变量n,所以此时n就不依赖于外部定义的i了
- 其实,就是当循环进行时,n这个局部变量会随着外部的i变化,这时候访问的时候第一个lambda中是0,第二个是1……以此类推,lambda对象中会存着对应n的结果,所以每次求值也就是能够得出相应的4和16的结果
其实这个问题不仅仅存在于lambda中,其他的操作中同理