首先我们先看下下面的代码
res = [lambda x : i*x for i in range(4)]
print([m(2) for m in res])
上述代码的执行结果是什么呢?很多人想的肯定是[0,2,4,6]吧?
实际的结果是这样的: [6,6,6,6]
为什么会出现这样的结果呢?
答案是:延迟绑定特性
首先我们看下面这个代码:
for i in range(10):
pass
print(i)
上面的代码结果为什么呢? 答案是9,相信很多人看到这里就会有种恍然大悟的感觉,原来是这样! 但具体是什么原理呢?请看下面的关于第一个代码的分析.
这个列表中有4个元素,每个元素都是一个匿名函数(实际上说是4个匿名函数也不完全准确,其实是4个匿名函数计算后的值,因为后面for i 的循环不光循环了4次,同时提还提供了i的变量引用,等待4次循环结束后,i指向一个值i=3,这个时候,匿名函数才开始引用i=3,计算结果。所以就会出现[6,6,6,6],因为匿名函数中的i并不是立即引用后面循环中的i值的,而是在运行嵌套函数的时候,才会查找i的值,这个特性也就是延迟绑定
如何得到我们需要的结果呢?
只需要这么写:
res = [lambda x,i=i : i*x for i in range(4)]
print([m(2) for m in res])
添加了一个i=i后,就给匿名函数,添加了一个默认参数,而python函数中的默认参数,是在python 解释器遇到def(i=i)或lambda 关键字时,就必须初始化默认参数,此时for i in range(4),每循环一次,匿名函数的默认参数i,就需要找一次i的引用,i=0时,第一个匿名函数的默认参数值就是0,i=1时,第二个匿名函数的默认参数值就是1,以此类推。