def multiply(): return [lambda x: i * x for i in range(4)] print([m(100) for m in multiply()]) [300, 300, 300, 300] 上面代碼的運(yùn)行結(jié)果很容易被誤判為[0, 100, 200, 300]。首先需要注意的是multiply函數(shù)用生成式語(yǔ)法返回了一個(gè)列表,列表中保存了4個(gè)Lambda函數(shù),這4個(gè)Lambda函數(shù)會(huì)返回傳入的參數(shù)乘以i的結(jié)果。需要注意的是這里有閉包(closure)現(xiàn)象,multiply函數(shù)中的局部變量i的生命周期被延展了,由于i最終的值是3,所以通過(guò)m(100)調(diào)列表中的Lambda函數(shù)時(shí)會(huì)返回300,而且4個(gè)調(diào)用都是如此。 如果想得到[0, 100, 200, 300]這個(gè)結(jié)果,可以按照下面幾種方式來(lái)修改multiply函數(shù)。
方法一:使用生成器,讓函數(shù)獲得i的當(dāng)前值。def multiply(): return (lambda x: i * x for i in range(4)) print([m(100) for m in multiply()]) def multiply(): for i in range(4): yield lambda x: x * i print([m(100) for m in multiply()])
方法二:使用偏函數(shù),徹底避開(kāi)閉包現(xiàn)象。 from functools import partial from operator import __mul__ def multiply(): return [partial(__mul__, i) for i in range(4)] print([m(100) for m in multiply()])