題目:
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
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/55136.html
標籤:Python
上一篇:Python命令列引數處理
