題目:
lst = [lambda x: x*i for i in range(4)] res = [m(2) for m in lst] print res
實際輸出:[6, 6, 6, 6]
想要輸出 [0, 2, 4, 6] 應該怎么改?如下:
lst = [lambda x, i=i: x*i for i in range(4)] res = [m(2) for m in lst] print res
這個問題涉及到了Python的閉包及延時綁定的知識(Python作用域)。
在Python核心編程里,閉包的定義如下:
如果在一個內部函數里,對外部作用域(但不是在全局作用域)的變量進行引用,那么內部函數就被認定是閉包。
總結為三點:
1、是一個內嵌函數
2、對外部函數變量引用
3、外部函數返回內嵌函數
簡單的閉包例子:
def counter(start_at=0): count = [start_at] def incr(): count[0] += 1 return count[0] return incr
上面的那道題,可以寫成這樣:
def func(): fun_list = [] for i in range(4): def foo(x): return x*i fun_list.append(foo) return fun_list for m in func(): print m(2)
func()是一個包含四個函數的列表:
[<function func at 0x00000000021CE9E8>, <function func at 0x00000000021CEA58>, <function func at 0x00000000021CEAC8>, <function func at 0x00000000021CEB38>]
當我們執行 m(2) 時,運行到foo()內部函數,發現變量 i 並不是foo()中的變量,於是就到外部函數func中尋找變量 i ,但此時外部的 for 已經循環完畢,最后的 i =3 。所以,每次
執行m(2),i 的值都是 3 ,因此,最終結果會是 [6, 6, 6, 6] 。
當在foo()中添加 i=i 后,即:
def func(): fun_list = [] for i in range(4): def foo(x, i=i): return x*i fun_list.append(foo) return fun_list for m in func(): print m(2)
這樣的話,for循環執行時,就已經把 i(0, 1, 2, 3) 的值傳給了foo()函數,此時的 i 已經是foo()函數的內部變量,運行到foo()函數時,就不會到外部函數尋找變量 i ,直接運行
x*i(0, 1, 2, 3),因此最終結果會是 [0, 2, 4, 6] 。