注:本文的原文是 5 Great Python Interview Questions,同時謝謝 @非烏龜 指出我的疏漏,沒有來源標記,也贊其細心,希望看文章的同時大家都能看下原文,因為每個人的理解不一致,原汁原味的最有幫助,我翻譯很多文章的目的一是為了自己以後找資料方便;二是作為一個索引,以後再看原文的時候,能更加快捷。其目的還是希望大家能看原文的。
問題一:以下的程式碼的輸出將是什麼? 說出你的答案並解釋。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
class Parent(object): x = 1 class Child1(Parent): pass class Child2(Parent): pass print Parent.x, Child1.x, Child2.x Child1.x = 2 print Parent.x, Child1.x, Child2.x Parent.x = 3 print Parent.x, Child1.x, Child2.x |
答案
以上程式碼的輸出是:
1 2 3 4 |
1 1 1 1 2 1 3 2 3 |
使你困惑或是驚奇的是關於最後一行的輸出是 3 2 3
而不是 3 2 1
。為什麼改變了 Parent.x
的值還會改變 Child2.x
的值,但是同時 Child1.x
值卻沒有改變?
這個答案的關鍵是,在 Python 中,類變數在內部是作為字典處理的。如果一個變數的名字沒有在當前類的字典中發現,將搜尋祖先類(比如父類)直到被引用的變數名被找到(如果這個被引用的變數名既沒有在自己所在的類又沒有在祖先類中找到,會引發一個 AttributeError
異常 )。
因此,在父類中設定 x = 1
會使得類變數 X
在引用該類和其任何子類中的值為 1。這就是因為第一個 print 語句的輸出是 1 1 1
。
隨後,如果任何它的子類重寫了該值(例如,我們執行語句 Child1.x = 2
),然後,該值僅僅在子類中被改變。這就是為什麼第二個 print
語句的輸出是 1 2 1
。
最後,如果該值在父類中被改變(例如,我們執行語句 Parent.x = 3
),這個改變會影響到任何未重寫該值的子類當中的值(在這個示例中被影響的子類是 Child2
)。這就是為什麼第三個 print
輸出是 3 2 3
。
問題二:以下的程式碼的輸出將是什麼? 說出你的答案並解釋?
1 2 3 4 5 6 7 8 9 10 11 |
def div1(x,y): print("%s/%s = %s" % (x, y, x/y)) def div2(x,y): print("%s//%s = %s" % (x, y, x//y)) div1(5,2) div1(5.,2) div2(5,2) div2(5.,2.) |
答案
這個答案實際依賴於你使用的是 Python 2 還是 Python 3。
在 Python 3 中,期望的輸出是:
1 2 3 4 5 |
5/2 = 2.5 5.0/2 = 2.5 5//2 = 2 5.0//2.0 = 2.0 |
在 Python 2 中,儘管如此,以上程式碼的輸出將是:
1 2 3 4 5 |
5/2 = 2 5.0/2 = 2.5 5//2 = 2 5.0//2.0 = 2.0 |
預設,如果兩個運算元都是整數,Python 2 自動執行整型計算。結果,5/2
值為 2
,然而 5./2
值為 2.5
。
注意,儘管如此,你可以在 Python 2 中過載這一行為(比如達到你想在 Python 3 中的同樣結果),通過新增以下匯入:
1 2 |
from __future__ import division |
也需要注意的是“雙劃線”(//)操作符將一直執行整除,而不管運算元的型別,這就是為什麼 5.0//2.0
值為 2.0
。
注: 在 Python 3 中,
/
操作符是做浮點除法,而//
是做整除(即商沒有餘數,比如 10 // 3 其結果就為 3,餘數會被截除掉,而(-7) // 3
的結果卻是-3
。這個演算法與其它很多程式語言不一樣,需要注意,它們的整除運算會向0的方向取值。而在 Python 2 中,/
就是整除,即和 Python 3 中的//
操作符一樣,)
問題三:以下程式碼將輸出什麼?
1 2 3 |
list = ['a', 'b', 'c', 'd', 'e'] print list[10:] |
答案
以上程式碼將輸出 []
,並且不會導致一個 IndexError
。
正如人們所期望的,試圖訪問一個超過列表索引值的成員將導致 IndexError
(比如訪問以上列表的 list[10]
)。儘管如此,試圖訪問一個列表的以超出列表成員數作為開始索引的切片將不會導致 IndexError
,並且將僅僅返回一個空列表。
一個討厭的小問題是它會導致出現 bug ,並且這個問題是難以追蹤的,因為它在執行時不會引發錯誤。
問題四:以下的程式碼的輸出將是什麼? 說出你的答案並解釋?
1 2 3 4 5 |
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
建立的函式展示的表現是一樣的。)
這裡有兩種方法解決這個問題。
最普遍的解決方案是建立一個閉包,通過使用預設引數立即繫結它的引數。例如:
1 2 3 |
def multipliers(): return [lambda x, i=i : i * x for i in range(4)] |
另外一個選擇是,你可以使用 functools.partial
函式:
1 2 3 4 5 6 |
from functools import partial from operator import mul def multipliers(): return [partial(mul, i) for i in range(4)] |
問題五:以下的程式碼的輸出將是什麼? 說出你的答案並解釋?
1 2 3 4 5 6 7 8 9 10 11 12 |
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
的定義來產生期望的結果
以上程式碼的輸出為:
1 2 3 4 |
list1 = [10, 'a'] list2 = [123] list3 = [10, 'a'] |
許多人會錯誤的認為 list1
應該等於 [10]
以及 list3
應該等於 ['a']
。認為 list
的引數會在 extendList
每次被呼叫的時候會被設定成它的預設值 []
。
儘管如此,實際發生的事情是,新的預設列表僅僅只在函式被定義時建立一次。隨後當 extendList
沒有被指定的列表引數呼叫的時候,其使用的是同一個列表。這就是為什麼當函式被定義的時候,表示式是用預設引數被計算,而不是它被呼叫的時候。
因此,list1
和 list3
是操作的相同的列表。而 list2
是操作的它建立的獨立的列表(通過傳遞它自己的空列表作為
list 引數的值)。
extendList
函式的定義可以做如下修改,但,當沒有新的 list
引數被指定的時候,會總是開始一個新列表,這更加可能是一直期望的行為。
1 2 3 4 5 6 |
def extendList(val, list=None): if list is None: list = [] list.append(val) return list |
使用這個改進的實現,輸出將是:
1 2 3 4 |
list1 = [10] list2 = [123] list3 = ['a'] |