問題四:以下的代碼的輸出將是什么? 說出你的答案并解釋?

def multipliers():return [lambda x : i * x for i in range(4)] print [m(2) for m in multipliers()] 

你將如何修改?multipliers?的定義來產生期望的結果

答案

以上代碼的輸出是?[6, 6, 6, 6]?(而不是?[0, 2, 4, 6])。

這個的原因是 Python 的閉包的后期綁定導致的?late binding,這意味著在閉包中的變量是在內部函數被調用的時候被查找。所以結果是,當任何?multipliers()?返回的函數被調用,在那時,i?的值是在它被調用時的周圍作用域中查找,到那時,無論哪個返回的函數被調用,for?循環都已經完成了,i?最后的值是?3,因此,每個返回的函數?multiplies?的值都是 3。因此一個等于 2 的值被傳遞進以上代碼,它們將返回一個值 6 (比如: 3 x 2)。

(順便說下,正如在?The Hitchhiker’s Guide to Python?中指出的,這里有一點普遍的誤解,是關于?lambda?表達式的一些東西。一個?lambda?表達式創建的函數不是特殊的,和使用一個普通的?def?創建的函數展示的表現是一樣的。)

這里有兩種方法解決這個問題。

最普遍的解決方案是創建一個閉包,通過使用默認參數立即綁定它的參數。例如:

def multipliers():return [lambda x, i=i : i * x for i in range(4)] 

另外一個選擇是,你可以使用?functools.partial?函數:

from functools import partial
from operator import muldef multipliers(): return [partial(mul, i) for i in range(4)] 

問題五:以下的代碼的輸出將是什么? 說出你的答案并解釋?

def extendList(val, list=[]):list.append(val)return list list1 = extendList(10) list2 = extendList(123,[]) list3 = extendList('a') print "list1 = %s" % list1 print "list2 = %s" % list2 print "list3 = %s" % list3 

你將如何修改?extendList?的定義來產生期望的結果

以上代碼的輸出為:

list1 = [10, 'a']
list2 = [123] list3 = [10, 'a'] 

許多人會錯誤的認為?list1?應該等于?[10]?以及?list3?應該等于?['a']。認為?list?的參數會在?extendList?每次被調用的時候會被設置成它的默認值?[]

盡管如此,實際發生的事情是,新的默認列表僅僅只在函數被定義時創建一次。隨后當?extendList?沒有被指定的列表參數調用的時候,其使用的是同一個列表。這就是為什么當函數被定義的時候,表達式是用默認參數被計算,而不是它被調用的時候。

因此,list1?和?list3?是操作的相同的列表。而 ````list2是操作的它創建的獨立的列表(通過傳遞它自己的空列表作為list``` 參數的值)。

extendList?函數的定義可以做如下修改,但,當沒有新的?list?參數被指定的時候,會總是開始一個新列表,這更加可能是一直期望的行為。

def extendList(val, list=None):if list is None: list = [] list.append(val) return list 

使用這個改進的實現,輸出將是:

list1 = [10]
list2 = [123] list3 = ['a']