Python程序員的30個(gè)常見(jiàn)錯誤
18. 不要試圖從那些會(huì )改變對象的函數得到結果
本文引用地址:http://dyxdggzs.com/article/201901/396507.htm諸如像方法list.append()和list.sort()一類(lèi)的直接改變操作會(huì )改變一個(gè)對象,但不會(huì )將它們改變的對象返回出來(lái)(它們會(huì )返回None);正確的做法是直接調用它們而不要將結果賦值。經(jīng)常會(huì )看見(jiàn)初學(xué)者會(huì )寫(xiě)諸如此類(lèi)的代碼:
mylist = mylist.append(X)
目的是要得到append的結果,但是事實(shí)上這樣做會(huì )將None賦值給mylist,而不是改變后的列表。更加特別的一個(gè)例子是想通過(guò)用排序后的鍵值來(lái)遍歷一個(gè)字典里的各個(gè)元素,請看下面的例子:
D = {...}
for k in D.keys().sort(): print D[k]
差一點(diǎn)兒就成功了——keys方法會(huì )創(chuàng )建一個(gè)keys的列表,然后用sort方法來(lái)將這個(gè)列表排序——但是因為sort方法會(huì )返回None,這個(gè)循環(huán)會(huì )失敗,因為它實(shí)際上是要遍歷None(這可不是一個(gè)序列)。要改正這段代碼,將方法的調用分離出來(lái),放在不同的語(yǔ)句中,如下:
Ks = D.keys()
Ks.sort()
for k in Ks: print D[k]
19. 只有在數字類(lèi)型中才存在類(lèi)型轉換
在Python中,一個(gè)諸如123+3.145的表達式是可以工作的——它會(huì )自動(dòng)將整數型轉換為浮點(diǎn)型,然后用浮點(diǎn)運算。但是下面的代碼就會(huì )出錯了:
S = "42"
I = 1
X = S + I # 類(lèi)型錯誤
這同樣也是有意而為的,因為這是不明確的:究竟是將字符串轉換為數字(進(jìn)行相加)呢,還是將數字轉換為字符串(進(jìn)行聯(lián)接)呢?在Python中,我們認為“明確比含糊好”(即,EIBTI(Explicit is better than implicit)),因此你得手動(dòng)轉換類(lèi)型:
X = int(S) + I # 做加法: 43
X = S + str(I) # 字符串聯(lián)接: "421"
20. 循環(huán)的數據結構會(huì )導致循環(huán)
盡管這在實(shí)際情況中很少見(jiàn),但是如果一個(gè)對象的集合包含了到它自己的引用,這被稱(chēng)為循環(huán)對象(cyclic object)。如果在一個(gè)對象中發(fā)現一個(gè)循環(huán),Python會(huì )輸出一個(gè)[…],以避免在無(wú)限循環(huán)中卡?。?/p>
>>> L = ['grail'] # 在 L中又引用L自身會(huì )
>>> L.append(L) # 在對象中創(chuàng )造一個(gè)循環(huán)
>>> L
['grail', [...]]
除了知道這三個(gè)點(diǎn)在對象中表示循環(huán)以外,這個(gè)例子也是很值得借鑒的。因為你可能無(wú)意間在你的代碼中出現這樣的循環(huán)的結構而導致你的代碼出錯。如果有必要的話(huà),維護一個(gè)列表或者字典來(lái)表示已經(jīng)訪(fǎng)問(wèn)過(guò)的對象,然后通過(guò)檢查它來(lái)確認你是否碰到了循環(huán)。
21. 賦值語(yǔ)句不會(huì )創(chuàng )建對象的副本,僅僅創(chuàng )建引用
這是Python的一個(gè)核心理念,有時(shí)候當行為不對時(shí)會(huì )帶來(lái)錯誤。在下面的例子中,一個(gè)列表對象被賦給了名為L(cháng)的變量,然后L又在列表M中被引用。內部改變L的話(huà),同時(shí)也會(huì )改變M所引用的對象,因為它們倆都指向同一個(gè)對象。
>>> L = [1, 2, 3] # 共用的列表對象
>>> M = ['X', L, 'Y'] # 嵌入一個(gè)到L的引用
>>> M
['X', [1, 2, 3], 'Y']
>>> L[1] = 0 # 也改變了M
>>> M
['X', [1, 0, 3], 'Y']
通常情況下只有在稍大一點(diǎn)的程序里這就顯得很重要了,而且這些共用的引用通常確實(shí)是你需要的。如果不是的話(huà),你可以明確的給他們創(chuàng )建一個(gè)副本來(lái)避免共用的引用;對于列表來(lái)說(shuō),你可以通過(guò)使用一個(gè)空列表的切片來(lái)創(chuàng )建一個(gè)頂層的副本:
>>> L = [1, 2, 3]
>>> M = ['X', L[:], 'Y'] # 嵌入一個(gè)L的副本
>>> L[1] = 0 # 僅僅改變了L,但是不影響M
>>> L
[1, 0, 3]
>>> M
['X', [1, 2, 3], 'Y']
切片的范圍起始從默認的0到被切片的序列的最大長(cháng)度。如果兩者都省略掉了,那么切片會(huì )抽取該序列中的所有元素,并創(chuàng )造一個(gè)頂層的副本(一個(gè)新的,不被公用的對象)。對于字典來(lái)說(shuō),使用字典的dict.copy()方法。
22. 靜態(tài)識別本地域的變量名
Python默認將一個(gè)函數中賦值的變量名視作是本地域的,它們存在于該函數的作用域中并且僅僅在函數運行的時(shí)候才存在。從技術(shù)上講,Python是在編譯def代碼時(shí),去靜態(tài)的識別本地變量,而不是在運行時(shí)碰到賦值的時(shí)候才識別到的。
如果不理解這點(diǎn)的話(huà),會(huì )引起人們的誤解。比如,看看下面的例子,當你在一個(gè)引用之后給一個(gè)變量賦值會(huì )怎么樣:
>>> X = 99
>>> def func():
... print X # 這個(gè)時(shí)候還不存在
... X = 88 # 在整個(gè)def中將X視作本地變量
...
>>> func( ) # 出錯了!
你會(huì )得到一個(gè)“未定義變量名”的錯誤,但是其原因是很微妙的。當編譯這則代碼時(shí),Python碰到給X賦值的語(yǔ)句時(shí)認為在這個(gè)函數中的任何地方X會(huì )被視作一個(gè)本地變量名。
但是之后當真正運行這個(gè)函數時(shí),執行print語(yǔ)句的時(shí)候,賦值語(yǔ)句還沒(méi)有發(fā)生,這樣Python便會(huì )報告一個(gè)“未定義變量名”的錯誤。
事實(shí)上,之前的這個(gè)例子想要做的事情是很模糊的:你是想要先輸出那個(gè)全局的X,然后創(chuàng )建一個(gè)本地的X呢,還是說(shuō)這是個(gè)程序的錯誤?如果你真的是想要輸出這個(gè)全局的X,你需要將它在一個(gè)全局語(yǔ)句中聲明它,或者通過(guò)包絡(luò )模塊的名字來(lái)引用它。
23. 默認參數和可變對象
在執行def語(yǔ)句時(shí),默認參數的值只被解析并保存一次,而不是每次在調用函數的時(shí)候。這通常是你想要的那樣,但是因為默認值需要在每次調用時(shí)都保持同樣對象,你在試圖改變可變的默認值(mutable defaults)的時(shí)候可要小心了。
例如,下面的函數中使用一個(gè)空的列表作為默認值,然后在之后每一次函數調用的時(shí)候改變它的值:
>>> def saver(x=[]): # 保存一個(gè)列表對象
... x.append(1) # 并每次調用的時(shí)候
... print x # 改變它的值
...
>>> saver([2]) # 未使用默認值
[2, 1]
>>> saver() # 使用默認值
[1]
>>> saver() # 每次調用都會(huì )增加!
[1, 1]
>>> saver()
[1, 1, 1]
有的人將這個(gè)視作Python的一個(gè)特點(diǎn)——因為可變的默認參數在每次函數調用時(shí)保持了它們的狀態(tài),它們能提供像C語(yǔ)言中靜態(tài)本地函數變量的類(lèi)似的一些功能。
但是,當你第一次碰到它時(shí)會(huì )覺(jué)得這很奇怪,并且在Python中有更加簡(jiǎn)單的辦法來(lái)在不同的調用之間保存狀態(tài)(比如說(shuō)類(lèi))。
要擺脫這樣的行為,在函數開(kāi)始的地方用切片或者方法來(lái)創(chuàng )建默認參數的副本,或者將默認值的表達式移到函數里面;只要每次函數調用時(shí)這些值在函數里,就會(huì )每次都得到一個(gè)新的對象:
>>> def saver(x=None):
... if x is None: x = [] # 沒(méi)有傳入參數?
... x.append(1) # 改變新的列表
... print x
...
>>> saver([2]) # 沒(méi)有使用默認值
[2, 1]
>>> saver() # 這次不會(huì )變了
[1]
>>> saver()
[1]
24. 其他常見(jiàn)的編程陷阱
下面列舉了其他的一些在這里沒(méi)法詳述的陷阱:
在頂層文件中語(yǔ)句的順序是有講究的:因為運行或者加載一個(gè)文件會(huì )從上到下運行它的語(yǔ)句,所以請確保將你未嵌套的函數調用或者類(lèi)的調用放在函數或者類(lèi)的定義之后。
reload不影響用from加載的名字:reload最好和import語(yǔ)句一起使用。如果你使用from語(yǔ)句,記得在reload之后重新運行一遍from,否則你仍然使用之前老的名字。
在多重繼承中混合的順序是有講究的:這是因為對superclass的搜索是從左到右的,在類(lèi)定義的頭部,在多重superclass中如果出現重復的名字,則以最左邊的類(lèi)名為準。
在t 888888 ry語(yǔ)句中空的except子句可能會(huì )比你預想的捕捉到更多的錯誤。在try語(yǔ)句中空的except子句表示捕捉所有的錯誤,即便是真正的程序錯誤,和sys.exit()調用,也會(huì )被捕捉到。
評論