在這篇文章中,我將總結(jié)新老Python程序員常犯的一些錯(cuò)誤钳宪,以幫助你們?cè)谧约旱墓ぷ鞅苊夥竿瑯踊蝾愃棋e(cuò)誤认臊。
首先我要說(shuō)明一下的是,這些都是來(lái)源于第一手的經(jīng)驗(yàn)吓懈。我以講授Python的知識(shí)為生歼冰。在過(guò)去的7年里,我已經(jīng)給上千名學(xué)生講授上百堂Python的課程骄瓣,同時(shí)看著這些學(xué)生們犯同樣的錯(cuò)。也就是說(shuō)耍攘,這些是我看著Python初學(xué)者活生生犯的錯(cuò)榕栏,千百次的錯(cuò)。
事實(shí)上蕾各,這些錯(cuò)誤實(shí)在是太普遍了以至于我敢保證你剛開始學(xué)的時(shí)候是一定會(huì)犯的扒磁。
“那么是什么呢?”你會(huì)問(wèn)式曲,“你也會(huì)在Python里犯那么多錯(cuò)么妨托?”是的。Python可能是最簡(jiǎn)單吝羞、最靈活的語(yǔ)言之一兰伤,但它終究還是一門編程語(yǔ)言。它仍然有語(yǔ)法钧排,數(shù)據(jù)類型敦腔,以及巫師蒂姆居住的黑暗角落。
典故出自《蒙蒂派森與圣杯》中的魔法師蒂姆恨溜,他主角們指點(diǎn)在洞穴的墻壁上記錄的圣杯位置符衔,作者在此處的意思是Python語(yǔ)言里容易犯錯(cuò)的地方。另糟袁,Python語(yǔ)言得名于作者Guido van Rossum特別喜歡的《蒙蒂派森飛行馬戲團(tuán)(Monty Python’s Flying Circus)》——譯者注
好事情是多虧了Python那干凈的設(shè)計(jì)判族,一旦你學(xué)會(huì)了Python,你就能自動(dòng)的避開很多陷阱项戴。Python在其各組件之間有著最小的互動(dòng)形帮,這能有效的減少bug。它也擁有十分簡(jiǎn)單的語(yǔ)法,這意味著在一開始你就有更小的概率犯錯(cuò)沃缘。當(dāng)你實(shí)在是犯了錯(cuò)的時(shí)候躯枢,Python的即時(shí)錯(cuò)誤檢測(cè)和報(bào)告能幫你迅速的恢復(fù)。
但用Python編程也不是個(gè)自動(dòng)完成的活兒槐臀,很多事還是要早做準(zhǔn)備锄蹂。那么廢話不多說(shuō)了,讓我們直切正題水慨。在接下來(lái)的三節(jié)里我們將這些錯(cuò)誤分為語(yǔ)用得糜、代碼,以及編程三個(gè)大類晰洒。
如果你想讀到更多的Python的常見(jiàn)錯(cuò)誤以及如何避免它們朝抖,那么在O’Reilly系列叢書的《Python學(xué)習(xí)手冊(cè)》(原書第5版)里有詳細(xì)的解讀。
號(hào):923414804
群里有志同道合的小伙伴谍珊,互幫互助治宣,
群里有不錯(cuò)的視頻學(xué)習(xí)教程和PDF!
語(yǔ)用錯(cuò)誤
讓我們從基礎(chǔ)開始砌滞,從那些剛學(xué)習(xí)編程的人鉆研語(yǔ)法之前碰到的事情開始侮邀。如果你已經(jīng)編過(guò)一些程了,那么以下這些可能看起來(lái)十分的簡(jiǎn)單贝润;如果你曾經(jīng)嘗試過(guò)教新手們?cè)趺淳幊贪砑耄鼈兛赡芫筒贿@么簡(jiǎn)單了。
1. 在交互提示符中輸入Python代碼
在>>>交互提示符中你只能輸入Python代碼打掘,而不是系統(tǒng)命令华畏。時(shí)常有人在這個(gè)提示符下輸入emacs,ls尊蚁,或者edit之類的命令亡笑,這些可不是Python代碼。
在Python代碼中確實(shí)有辦法來(lái)調(diào)用系統(tǒng)命令(例如os.system和os.popen)横朋,但可不是像直接輸入命令這么直接况芒。如果你想要在交互提示符中啟動(dòng)一個(gè)Python文件,請(qǐng)用import file叶撒,而不是系統(tǒng)命令python file.py绝骚。
2. Print語(yǔ)句(僅僅)是在文件中需要
因?yàn)榻换ソ忉屍鲿?huì)自動(dòng)的講表達(dá)式的結(jié)果輸出,所以你不需要交互的鍵入完整的print語(yǔ)句祠够。這是個(gè)很棒的功能压汪,但是記住在代碼文件里,通常你只有用print語(yǔ)句才能看得到輸出古瓤。
3. 小心Windows里的自動(dòng)擴(kuò)展名
如果你在Windows里使用記事本來(lái)編輯代碼文件的話止剖,當(dāng)你保持的時(shí)候小心選擇“所有文件”(All Files)這個(gè)類型腺阳,并且明確的給你的文件加一個(gè).py的后綴。不然的話記事本會(huì)給你的文件加一個(gè).txt的擴(kuò)展名穿香,使得在某些啟動(dòng)方法中沒(méi)法跑這個(gè)程序亭引。
更糟糕的是,像Word或者是寫字板一類的文字處理軟件還會(huì)默認(rèn)的加上一些格式字符皮获,而這些字符Python語(yǔ)法是不認(rèn)的焙蚓。
所以記得,在Windows下總是選“所有文件”(All Files)洒宝,并保存為純文本购公,或者使用更加“編程友好”的文本編輯工具,比如IDLE雁歌。在IDLE中宏浩,記得在保存時(shí)手動(dòng)加上.py的擴(kuò)展名。
4. 在Windows下點(diǎn)擊圖標(biāo)的問(wèn)題
在Windows下靠瞎,你能靠點(diǎn)擊Python文件來(lái)啟動(dòng)一個(gè)Python程序比庄,但這有時(shí)會(huì)有問(wèn)題。首先乏盐,程序的輸出窗口在程序結(jié)束的瞬間也就消失了佳窑,要讓它不消失,你可以在文件最后加一條raw_input()的調(diào)用丑勤。另外华嘹,記住如果有錯(cuò)的話吧趣,輸出窗口也就立即消失了法竞。
要看到你的錯(cuò)誤信息的話,用別的方法來(lái)調(diào)用你的程序:比如從系統(tǒng)命令行啟動(dòng)强挫,通過(guò)提示符下用import語(yǔ)句岔霸,或者IDLE菜單里的選項(xiàng),等等俯渤。
5. Import只在第一次有效
你可以在交互提示符中通過(guò)import一個(gè)文件來(lái)運(yùn)行它呆细,但是這只會(huì)在一個(gè)會(huì)話中起一次作用;接下來(lái)的import僅僅是返回這個(gè)已經(jīng)加載的模塊八匠。要想強(qiáng)制Python重新加載一個(gè)文件的代碼絮爷,請(qǐng)調(diào)用函數(shù)reload(module)來(lái)達(dá)到這個(gè)目的。注意對(duì)reload請(qǐng)使用括號(hào)梨树,而import不要使用括號(hào)坑夯。
6. 空白行(僅僅)在交互提示符中有作用
在模塊文件中空白行和注釋統(tǒng)統(tǒng)會(huì)被忽略掉,但是在交互提示符中鍵入代碼時(shí)抡四,空白行表示一個(gè)復(fù)合語(yǔ)句的結(jié)束柜蜈。
換句話說(shuō)仗谆,空白行告訴交互提示符你完成了一個(gè)復(fù)合語(yǔ)句;在你真正完成之前不要鍵入回車淑履。事實(shí)上當(dāng)你要開始一個(gè)新的語(yǔ)句時(shí)隶垮,你需要鍵入一個(gè)空行來(lái)結(jié)束當(dāng)前的語(yǔ)句——交互提示符一次只運(yùn)行一條語(yǔ)句。
02 代碼錯(cuò)誤
一旦你開始認(rèn)真寫Python代碼了秘噪,接下來(lái)了一堆陷阱就更加危險(xiǎn)了——這些都是一些跨語(yǔ)言特性的基本代碼錯(cuò)誤狸吞,并常常困擾不細(xì)心的程序員。
7. 別忘了冒號(hào)
這是新手程序員最容易犯的一個(gè)錯(cuò)誤:別忘了在復(fù)合語(yǔ)句的起始語(yǔ)句(if缆娃,while捷绒, for等語(yǔ)句的第一行)結(jié)束的地方加上一個(gè)冒號(hào)“:”。也許你剛開始會(huì)忘掉這個(gè)贯要,但是到了很快這就會(huì)成為一個(gè)下意識(shí)的習(xí)慣暖侨。課堂里75%的學(xué)生當(dāng)天就可以記住這個(gè)。
8. 初始化變量
在Python里崇渗,一個(gè)表達(dá)式中的名字在它被賦值之前是沒(méi)法使用的字逗。這是有意而為的:這樣能避免一些輸入失誤,同時(shí)也能避免默認(rèn)究竟應(yīng)該是什么類型的問(wèn)題(0宅广,None葫掉,””,[]跟狱,?)俭厚。記住把計(jì)數(shù)器初始化為0,列表初始化為[]驶臊,以此類推挪挤。
9. 從第一列開始
確保把頂層的,未嵌套的代碼放在最左邊第一列開始关翎。這包括在模塊文件中未嵌套的代碼扛门,以及在交互提示符中未嵌套的代碼。Python使用縮進(jìn)的辦法來(lái)區(qū)分嵌套的代碼段纵寝,因此在你代碼左邊的空格意味著嵌套的代碼塊论寨。除了縮進(jìn)以外,空格通常是被忽略掉的爽茴。
10. 縮進(jìn)一致
在同一個(gè)代碼塊中避免講tab和空格混用來(lái)縮進(jìn)葬凳,除非你知道運(yùn)行你的代碼的系統(tǒng)是怎么處理tab的。否則的話室奏,在你的編輯器里看起來(lái)是tab的縮進(jìn)也許Python看起來(lái)就會(huì)被視作是一些空格火焰。保險(xiǎn)起見(jiàn),在每個(gè)代碼塊中全都是用tab或者全都是用空格來(lái)縮進(jìn)窍奋;用多少由你決定荐健。
11. 在函數(shù)調(diào)用時(shí)使用括號(hào)
無(wú)論一個(gè)函數(shù)是否需要參數(shù)酱畅,你必須要加一對(duì)括號(hào)來(lái)調(diào)用它。即江场,使用function()纺酸,而不是function。Python的函數(shù)簡(jiǎn)單來(lái)說(shuō)是具有特殊功能(調(diào)用)的對(duì)象址否,而調(diào)用是用括號(hào)來(lái)觸發(fā)的餐蔬。像所有的對(duì)象一樣,他們也可以被賦值給變量佑附,并且間接的使用他們:x=function:x()樊诺。在Python的培訓(xùn)中,這樣的錯(cuò)誤常常在文件的操作中出現(xiàn)音同。通常會(huì)看到新手用file.close來(lái)關(guān)閉一個(gè)問(wèn)題词爬,而不是用file.close()。因?yàn)樵赑ython中引用一個(gè)函數(shù)而不調(diào)用它是合法的权均,因此不使用括號(hào)的操作(file.close)無(wú)聲的成功了顿膨,但是并沒(méi)有關(guān)閉這個(gè)文件!
12. 在Import時(shí)不要使用表達(dá)式或者路徑
在系統(tǒng)的命令行里使用文件夾路徑或者文件的擴(kuò)展名叽赊,但不要在import語(yǔ)句中使用恋沃。即,使用import mod必指,而不是import mod.py囊咏,或者import dir/mod.py。
在實(shí)際情況中塔橡,這大概是初學(xué)者常犯的第二大錯(cuò)誤了梅割。因?yàn)槟K會(huì)有除了.py以為的其他的后綴(例如,.pyc)谱邪,強(qiáng)制寫上某個(gè)后綴不僅是不合語(yǔ)法的炮捧,也沒(méi)有什么意義庶诡。和系統(tǒng)有關(guān)的目錄路徑的格式是從你的模塊搜索路徑的設(shè)置里來(lái)的惦银,而不是import語(yǔ)句。你可以在文件名里使用點(diǎn)來(lái)指向包的子目錄(例如末誓,import dir1.dir2.mod)扯俱,但是最左邊的目錄必須得通過(guò)模塊搜索路徑能夠找到,并且沒(méi)有在import中沒(méi)有其他路徑格式喇澡。
不正確的語(yǔ)句import mod.py被Python認(rèn)為是要記在一個(gè)包迅栅,它先加載一個(gè)模塊mod,然后試圖通過(guò)在一個(gè)叫做mod的目錄里去找到叫做py的模塊晴玖,最后可能什么也找不到而報(bào)出一系列費(fèi)解的錯(cuò)誤信息读存。
13. 不要在Python中寫C代碼
以下是給不熟悉Python的C程序員的一些備忘貼士:
- 在if和while中條件測(cè)試時(shí)为流,不用輸入括號(hào)(例如,if (X==1):)让簿。如果你喜歡的話敬察,加上括號(hào)也無(wú)妨,只是在這里是完全多余的尔当。
- 不要用分號(hào)來(lái)結(jié)束你的語(yǔ)句莲祸。從技術(shù)上講這在Python里是合法的,但是這毫無(wú)用處椭迎,除非你要把很多語(yǔ)句放在同一行里(例如锐帜,x=1; y=2; z=3)。
- 不要在while循環(huán)的條件測(cè)試中嵌入賦值語(yǔ)句(例如畜号,while ((x=next() != NULL))缴阎。在Python中,需要表達(dá)式的地方不能出現(xiàn)語(yǔ)句简软,并且賦值語(yǔ)句不是一個(gè)表達(dá)式药蜻。
03 編程錯(cuò)誤
下面終于要講到當(dāng)你用到更多的Python的功能(數(shù)據(jù)類型,函數(shù)替饿,模塊语泽,類等等)時(shí)可能碰到的問(wèn)題了。由于篇幅有限视卢,這里盡量精簡(jiǎn)踱卵,尤其是對(duì)一些高級(jí)的概念。要想了解更多的細(xì)節(jié)据过,敬請(qǐng)閱讀《Python學(xué)習(xí)手冊(cè)》惋砂。
14. 打開文件的調(diào)用不使用模塊搜索路徑
當(dāng)你在Python中調(diào)用open()來(lái)訪問(wèn)一個(gè)外部的文件時(shí),Python不會(huì)使用模塊搜索路徑來(lái)定位這個(gè)目標(biāo)文件绳锅。它會(huì)使用你提供的絕對(duì)路徑西饵,或者假定這個(gè)文件是在當(dāng)前工作目錄中。模塊搜索路徑僅僅為模塊加載服務(wù)的鳞芙。
15. 不同的類型對(duì)應(yīng)的方法也不同
列表的方法是不能用在字符串上的眷柔,反之亦然。通常情況下原朝,方法的調(diào)用是和數(shù)據(jù)類型有關(guān)的驯嘱,但是內(nèi)部函數(shù)通常在很多類型上都可以使用。舉個(gè)例子來(lái)說(shuō)喳坠,列表的reverse方法僅僅對(duì)列表有用鞠评,但是len函數(shù)對(duì)任何具有長(zhǎng)度的對(duì)象都適用。
16. 不能直接改變不可變數(shù)據(jù)類型
記住你沒(méi)法直接的改變一個(gè)不可變的對(duì)象(例如壕鹉,元組剃幌,字符串):
T = (1, 2, 3)
T[2] = 4 # 錯(cuò)誤
用切片聋涨,聯(lián)接等構(gòu)建一個(gè)新的對(duì)象,并根據(jù)需求將原來(lái)變量的值賦給它负乡。因?yàn)镻ython會(huì)自動(dòng)回收沒(méi)有用的內(nèi)存牛郑,因此這沒(méi)有看起來(lái)那么浪費(fèi):
T = T[:2] + (4,) # 沒(méi)問(wèn)題了: T 變成了 (1, 2, 4)
17. 使用簡(jiǎn)單的for循環(huán)而不是while或者range
當(dāng)你要從左到右遍歷一個(gè)有序的對(duì)象的所有元素時(shí),用簡(jiǎn)單的for循環(huán)(例如敬鬓,for x in seq:)相比于基于while-或者range-的計(jì)數(shù)循環(huán)而言會(huì)更容易寫淹朋,通常運(yùn)行起來(lái)也更快。
除非你一定需要钉答,盡量避免在一個(gè)for循環(huán)里使用range:讓Python來(lái)替你解決標(biāo)號(hào)的問(wèn)題础芍。在下面的例子中三個(gè)循環(huán)結(jié)構(gòu)都沒(méi)有問(wèn)題,但是第一個(gè)通常來(lái)說(shuō)更好数尿;在Python里仑性,簡(jiǎn)單至上。
S = "lumberjack"
for c in S: print c # 最簡(jiǎn)單
for i in range(len(S)): print S[i] # 太多了
i = 0 # 太多了
while i < len(S): print S[i]; i += 1
18. 不要試圖從那些會(huì)改變對(duì)象的函數(shù)得到結(jié)果
諸如像方法list.append()和list.sort()一類的直接改變操作會(huì)改變一個(gè)對(duì)象右蹦,但不會(huì)將它們改變的對(duì)象返回出來(lái)(它們會(huì)返回None)诊杆;正確的做法是直接調(diào)用它們而不要將結(jié)果賦值。經(jīng)常會(huì)看見(jiàn)初學(xué)者會(huì)寫諸如此類的代碼:
mylist = mylist.append(X)
目的是要得到append的結(jié)果何陆,但是事實(shí)上這樣做會(huì)將None賦值給mylist晨汹,而不是改變后的列表。更加特別的一個(gè)例子是想通過(guò)用排序后的鍵值來(lái)遍歷一個(gè)字典里的各個(gè)元素贷盲,請(qǐng)看下面的例子:
D = {...}
for k in D.keys().sort(): print D[k]
差一點(diǎn)兒就成功了——keys方法會(huì)創(chuàng)建一個(gè)keys的列表淘这,然后用sort方法來(lái)將這個(gè)列表排序——但是因?yàn)閟ort方法會(huì)返回None,這個(gè)循環(huán)會(huì)失敗巩剖,因?yàn)樗鼘?shí)際上是要遍歷None(這可不是一個(gè)序列)铝穷。要改正這段代碼,將方法的調(diào)用分離出來(lái)佳魔,放在不同的語(yǔ)句中曙聂,如下:
Ks = D.keys()
Ks.sort()
for k in Ks: print D[k]
19. 只有在數(shù)字類型中才存在類型轉(zhuǎn)換
在Python中,一個(gè)諸如123+3.145的表達(dá)式是可以工作的——它會(huì)自動(dòng)將整數(shù)型轉(zhuǎn)換為浮點(diǎn)型鞠鲜,然后用浮點(diǎn)運(yùn)算宁脊。但是下面的代碼就會(huì)出錯(cuò)了:
S = "42"
I = 1
X = S + I # 類型錯(cuò)誤
這同樣也是有意而為的,因?yàn)檫@是不明確的:究竟是將字符串轉(zhuǎn)換為數(shù)字(進(jìn)行相加)呢镊尺,還是將數(shù)字轉(zhuǎn)換為字符串(進(jìn)行聯(lián)接)呢朦佩?在Python中并思,我們認(rèn)為“明確比含糊好”(即庐氮,EIBTI(Explicit is better than implicit)),因此你得手動(dòng)轉(zhuǎn)換類型:
X = int(S) + I # 做加法: 43
X = S + str(I) # 字符串聯(lián)接: "421"
20. 循環(huán)的數(shù)據(jù)結(jié)構(gòu)會(huì)導(dǎo)致循環(huán)
盡管這在實(shí)際情況中很少見(jiàn)宋彼,但是如果一個(gè)對(duì)象的集合包含了到它自己的引用,這被稱為循環(huán)對(duì)象(cyclic object)。如果在一個(gè)對(duì)象中發(fā)現(xiàn)一個(gè)循環(huán)邻吞,Python會(huì)輸出一個(gè)[…]竞慢,以避免在無(wú)限循環(huán)中卡住:
>>> L = [ grail ] # 在 L中又引用L自身會(huì)
>>> L.append(L) # 在對(duì)象中創(chuàng)造一個(gè)循環(huán)
>>> L
[ grail , [...]]
除了知道這三個(gè)點(diǎn)在對(duì)象中表示循環(huán)以外炒俱,這個(gè)例子也是很值得借鑒的。因?yàn)槟憧赡軣o(wú)意間在你的代碼中出現(xiàn)這樣的循環(huán)的結(jié)構(gòu)而導(dǎo)致你的代碼出錯(cuò)。如果有必要的話寸士,維護(hù)一個(gè)列表或者字典來(lái)表示已經(jīng)訪問(wèn)過(guò)的對(duì)象,然后通過(guò)檢查它來(lái)確認(rèn)你是否碰到了循環(huán)碴卧。
21. 賦值語(yǔ)句不會(huì)創(chuàng)建對(duì)象的副本弱卡,僅僅創(chuàng)建引用
這是Python的一個(gè)核心理念,有時(shí)候當(dāng)行為不對(duì)時(shí)會(huì)帶來(lái)錯(cuò)誤住册。在下面的例子中婶博,一個(gè)列表對(duì)象被賦給了名為L(zhǎng)的變量,然后L又在列表M中被引用荧飞。內(nèi)部改變L的話凡人,同時(shí)也會(huì)改變M所引用的對(duì)象,因?yàn)樗鼈儌z都指向同一個(gè)對(duì)象叹阔。
>>> L = [1, 2, 3] # 共用的列表對(duì)象
>>> 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í)是你需要的。如果不是的話耳幢,你可以明確的給他們創(chuàng)建一個(gè)副本來(lái)避免共用的引用忠荞;對(duì)于列表來(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 ]
切片的范圍起始從默認(rèn)的0到被切片的序列的最大長(zhǎng)度委煤。如果兩者都省略掉了,那么切片會(huì)抽取該序列中的所有元素修档,并創(chuàng)造一個(gè)頂層的副本(一個(gè)新的碧绞,不被公用的對(duì)象)。對(duì)于字典來(lái)說(shuō)吱窝,使用字典的dict.copy()方法讥邻。
22. 靜態(tài)識(shí)別本地域的變量名
Python默認(rèn)將一個(gè)函數(shù)中賦值的變量名視作是本地域的,它們存在于該函數(shù)的作用域中并且僅僅在函數(shù)運(yùn)行的時(shí)候才存在院峡。從技術(shù)上講兴使,Python是在編譯def代碼時(shí),去靜態(tài)的識(shí)別本地變量照激,而不是在運(yùn)行時(shí)碰到賦值的時(shí)候才識(shí)別到的发魄。
如果不理解這點(diǎn)的話,會(huì)引起人們的誤解。比如励幼,看看下面的例子汰寓,當(dāng)你在一個(gè)引用之后給一個(gè)變量賦值會(huì)怎么樣:
>>> X = 99
>>> def func():
... print X # 這個(gè)時(shí)候還不存在
... X = 88 # 在整個(gè)def中將X視作本地變量
...
>>> func( ) # 出錯(cuò)了!
你會(huì)得到一個(gè)“未定義變量名”的錯(cuò)誤苹粟,但是其原因是很微妙的有滑。當(dāng)編譯這則代碼時(shí),Python碰到給X賦值的語(yǔ)句時(shí)認(rèn)為在這個(gè)函數(shù)中的任何地方X會(huì)被視作一個(gè)本地變量名嵌削。
但是之后當(dāng)真正運(yùn)行這個(gè)函數(shù)時(shí)毛好,執(zhí)行print語(yǔ)句的時(shí)候,賦值語(yǔ)句還沒(méi)有發(fā)生苛秕,這樣Python便會(huì)報(bào)告一個(gè)“未定義變量名”的錯(cuò)誤睛榄。
事實(shí)上,之前的這個(gè)例子想要做的事情是很模糊的:你是想要先輸出那個(gè)全局的X想帅,然后創(chuàng)建一個(gè)本地的X呢场靴,還是說(shuō)這是個(gè)程序的錯(cuò)誤?如果你真的是想要輸出這個(gè)全局的X港准,你需要將它在一個(gè)全局語(yǔ)句中聲明它旨剥,或者通過(guò)包絡(luò)模塊的名字來(lái)引用它。
23. 默認(rèn)參數(shù)和可變對(duì)象
在執(zhí)行def語(yǔ)句時(shí)浅缸,默認(rèn)參數(shù)的值只被解析并保存一次轨帜,而不是每次在調(diào)用函數(shù)的時(shí)候。這通常是你想要的那樣衩椒,但是因?yàn)槟J(rèn)值需要在每次調(diào)用時(shí)都保持同樣對(duì)象蚌父,你在試圖改變可變的默認(rèn)值(mutable defaults)的時(shí)候可要小心了。
例如毛萌,下面的函數(shù)中使用一個(gè)空的列表作為默認(rèn)值苟弛,然后在之后每一次函數(shù)調(diào)用的時(shí)候改變它的值:
>>> def saver(x=[]): # 保存一個(gè)列表對(duì)象
... x.append(1) # 并每次調(diào)用的時(shí)候
... print x # 改變它的值
...
>>> saver([2]) # 未使用默認(rèn)值
[2, 1]
>>> saver() # 使用默認(rèn)值
[1]
>>> saver() # 每次調(diào)用都會(huì)增加!
[1, 1]
>>> saver()
[1, 1, 1]
有的人將這個(gè)視作Python的一個(gè)特點(diǎn)——因?yàn)榭勺兊哪J(rèn)參數(shù)在每次函數(shù)調(diào)用時(shí)保持了它們的狀態(tài)阁将,它們能提供像C語(yǔ)言中靜態(tài)本地函數(shù)變量的類似的一些功能膏秫。
但是,當(dāng)你第一次碰到它時(shí)會(huì)覺(jué)得這很奇怪做盅,并且在Python中有更加簡(jiǎn)單的辦法來(lái)在不同的調(diào)用之間保存狀態(tài)(比如說(shuō)類)缤削。
要擺脫這樣的行為,在函數(shù)開始的地方用切片或者方法來(lái)創(chuàng)建默認(rèn)參數(shù)的副本吹榴,或者將默認(rèn)值的表達(dá)式移到函數(shù)里面亭敢;只要每次函數(shù)調(diào)用時(shí)這些值在函數(shù)里,就會(huì)每次都得到一個(gè)新的對(duì)象:
>>> def saver(x=None):
... if x is None: x = [] # 沒(méi)有傳入?yún)?shù)图筹?
... x.append(1) # 改變新的列表
... print x
...
>>> saver([2]) # 沒(méi)有使用默認(rèn)值
[2, 1]
>>> saver() # 這次不會(huì)變了
[1]
>>> saver()
[1]
24. 其他常見(jiàn)的編程陷阱
下面列舉了其他的一些在這里沒(méi)法詳述的陷阱:
- 在頂層文件中語(yǔ)句的順序是有講究的:因?yàn)檫\(yùn)行或者加載一個(gè)文件會(huì)從上到下運(yùn)行它的語(yǔ)句帅刀,所以請(qǐng)確保將你未嵌套的函數(shù)調(diào)用或者類的調(diào)用放在函數(shù)或者類的定義之后。
- reload不影響用from加載的名字:reload最好和import語(yǔ)句一起使用。如果你使用from語(yǔ)句劝篷,記得在reload之后重新運(yùn)行一遍from哨鸭,否則你仍然使用之前老的名字民宿。
- 在多重繼承中混合的順序是有講究的:這是因?yàn)閷?duì)superclass的搜索是從左到右的娇妓,在類定義的頭部,在多重superclass中如果出現(xiàn)重復(fù)的名字活鹰,則以最左邊的類名為準(zhǔn)哈恰。
- 在try語(yǔ)句中空的except子句可能會(huì)比你預(yù)想的捕捉到更多的錯(cuò)誤。在try語(yǔ)句中空的except子句表示捕捉所有的錯(cuò)誤志群,即便是真正的程序錯(cuò)誤着绷,和sys.exit()調(diào)用,也會(huì)被捕捉到锌云。
- 兔子可能會(huì)比他們看起來(lái)更加危險(xiǎn)荠医。