關於 [lambda x: x*i for i in range(4)] 理解


題目:

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] 。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM