題目:
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
上面的那道題,可以寫成這樣:
'''
遇到問題沒人解答?小編創建了一個Python學習交流QQ群:778463939
尋找有志同道合的小伙伴,互幫互助,群里還有不錯的視頻學習教程和PDF電子書!
'''
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] 。
轉自:https://blog.csdn.net/qdPython/article/details/107938206