文章中,我將總結(jié)新老Python程序員常犯的一些錯誤,以幫助你們在自己的工作避免犯同樣或類似錯誤。
首先我要說明一下的是,這些都是來源于第一手的經(jīng)驗。我以講授Python的知識為生。在過去的7年里,我已經(jīng)給上千名學(xué)生講授上百堂Python的課程,同時看著這些學(xué)生們犯同樣的錯。也就是說,這些是我看著Python初學(xué)者活生生犯的錯,千百次的錯。
事實上,這些錯誤實在是太普遍了以至于我敢保證你剛開始學(xué)的時候是一定會犯的。
“那么是什么呢?”你會問,“你也會在Python里犯那么多錯么?”是的。Python可能是最簡單、最靈活的語言之一,但它終究還是一門編程語言。它仍然有語法,數(shù)據(jù)類型,以及巫師蒂姆居住的黑暗角落。
典故出自《蒙蒂派森與圣杯》中的魔法師蒂姆,他主角們指點(diǎn)在洞穴的墻壁上記錄的圣杯位置,作者在此處的意思是Python語言里容易犯錯的地方。另,Python語言得名于作者Guido van Rossum特別喜歡的《蒙蒂派森飛行馬戲團(tuán)(Monty Python’s Flying Circus)》——譯者注
好事情是多虧了Python那干凈的設(shè)計,一旦你學(xué)會了Python,你就能自動的避開很多陷阱。Python在其各組件之間有著最小的互動,這能有效的減少bug。它也擁有十分簡單的語法,這意味著在一開始你就有更小的概率犯錯。當(dāng)你實在是犯了錯的時候,Python的即時錯誤檢測和報告能幫你迅速的恢復(fù)。
但用Python編程也不是個自動完成的活兒,很多事還是要早做準(zhǔn)備。那么廢話不多說了,讓我們直切正題。在接下來的三節(jié)里我們將這些錯誤分為語用、代碼,以及編程三個大類。
如果你想讀到更多的Python的常見錯誤以及如何避免它們,那么在O’Reilly系列叢書的《Python學(xué)習(xí)手冊》(原書第5版)里有詳細(xì)的解讀。
01 語用錯誤
讓我們從基礎(chǔ)開始,從那些剛學(xué)習(xí)編程的人鉆研語法之前碰到的事情開始。如果你已經(jīng)編過一些程了,那么以下這些可能看起來十分的簡單;如果你曾經(jīng)嘗試過教新手們怎么編程,它們可能就不這么簡單了。
1. 在交互提示符中輸入Python代碼
在>>>交互提示符中你只能輸入Python代碼,而不是系統(tǒng)命令。時常有人在這個提示符下輸入emacs,ls,或者edit之類的命令,這些可不是Python代碼。
在Python代碼中確實有辦法來調(diào)用系統(tǒng)命令(例如os.system和os.popen),但可不是像直接輸入命令這么直接。如果你想要在交互提示符中啟動一個Python文件,請用import file,而不是系統(tǒng)命令python file.py。
2. Print語句(僅僅)是在文件中需要
因為交互解釋器會自動的講表達(dá)式的結(jié)果輸出,所以你不需要交互的鍵入完整的print語句。這是個很棒的功能,但是記住在代碼文件里,通常你只有用print語句才能看得到輸出。
3. 小心Windows里的自動擴(kuò)展名
如果你在Windows里使用記事本來編輯代碼文件的話,當(dāng)你保持的時候小心選擇“所有文件”(All Files)這個類型,并且明確的給你的文件加一個.py的后綴。不然的話記事本會給你的文件加一個.txt的擴(kuò)展名,使得在某些啟動方法中沒法跑這個程序。
更糟糕的是,像Word或者是寫字板一類的文字處理軟件還會默認(rèn)的加上一些格式字符,而這些字符Python語法是不認(rèn)的。
所以記得,在Windows下總是選“所有文件”(All Files),并保存為純文本,或者使用更加“編程友好”的文本編輯工具,比如IDLE。在IDLE中,記得在保存時手動加上.py的擴(kuò)展名。
4. 在Windows下點(diǎn)擊圖標(biāo)的問題
在Windows下,你能靠點(diǎn)擊Python文件來啟動一個Python程序,但這有時會有問題。首先,程序的輸出窗口在程序結(jié)束的瞬間也就消失了,要讓它不消失,你可以在文件最后加一條raw_input()的調(diào)用。另外,記住如果有錯的話,輸出窗口也就立即消失了。
要看到你的錯誤信息的話,用別的方法來調(diào)用你的程序:比如從系統(tǒng)命令行啟動,通過提示符下用import語句,或者IDLE菜單里的選項,等等。
5. Import只在第一次有效
你可以在交互提示符中通過import一個文件來運(yùn)行它,但是這只會在一個會話中起一次作用;接下來的import僅僅是返回這個已經(jīng)加載的模塊。要想強(qiáng)制Python重新加載一個文件的代碼,請調(diào)用函數(shù)reload(module)來達(dá)到這個目的。注意對reload請使用括號,而import不要使用括號。
6. 空白行(僅僅)在交互提示符中有作用
在模塊文件中空白行和注釋統(tǒng)統(tǒng)會被忽略掉,但是在交互提示符中鍵入代碼時,空白行表示一個復(fù)合語句的結(jié)束。
換句話說,空白行告訴交互提示符你完成了一個復(fù)合語句;在你真正完成之前不要鍵入回車。事實上當(dāng)你要開始一個新的語句時,你需要鍵入一個空行來結(jié)束當(dāng)前的語句——交互提示符一次只運(yùn)行一條語句。
02 代碼錯誤
一旦你開始認(rèn)真寫Python代碼了,接下來了一堆陷阱就更加危險了——這些都是一些跨語言特性的基本代碼錯誤,并常常困擾不細(xì)心的程序員。
7. 別忘了冒號
這是新手程序員最容易犯的一個錯誤:別忘了在復(fù)合語句的起始語句(if,while, for等語句的第一行)結(jié)束的地方加上一個冒號“:”。也許你剛開始會忘掉這個,但是到了很快這就會成為一個下意識的習(xí)慣。課堂里75%的學(xué)生當(dāng)天就可以記住這個。
8. 初始化變量
在Python里,一個表達(dá)式中的名字在它被賦值之前是沒法使用的。這是有意而為的:這樣能避免一些輸入失誤,同時也能避免默認(rèn)究竟應(yīng)該是什么類型的問題(0,None,””,[],?)。記住把計數(shù)器初始化為0,列表初始化為[],以此類推。
9. 從第一列開始
確保把頂層的,未嵌套的代碼放在最左邊第一列開始。這包括在模塊文件中未嵌套的代碼,以及在交互提示符中未嵌套的代碼。Python使用縮進(jìn)的辦法來區(qū)分嵌套的代碼段,因此在你代碼左邊的空格意味著嵌套的代碼塊。除了縮進(jìn)以外,空格通常是被忽略掉的。
10. 縮進(jìn)一致
在同一個代碼塊中避免講tab和空格混用來縮進(jìn),除非你知道運(yùn)行你的代碼的系統(tǒng)是怎么處理tab的。否則的話,在你的編輯器里看起來是tab的縮進(jìn)也許Python看起來就會被視作是一些空格。保險起見,在每個代碼塊中全都是用tab或者全都是用空格來縮進(jìn);用多少由你決定。
11. 在函數(shù)調(diào)用時使用括號
無論一個函數(shù)是否需要參數(shù),你必須要加一對括號來調(diào)用它。即,使用function(),而不是function。Python的函數(shù)簡單來說是具有特殊功能(調(diào)用)的對象,而調(diào)用是用括號來觸發(fā)的。像所有的對象一樣,他們也可以被賦值給變量,并且間接的使用他們:x=function:x()。
在Python的培訓(xùn)中,這樣的錯誤常常在文件的操作中出現(xiàn)。通常會看到新手用file.close來關(guān)閉一個問題,而不是用file.close()。因為在Python中引用一個函數(shù)而不調(diào)用它是合法的,因此不使用括號的操作(file.close)無聲的成功了,但是并沒有關(guān)閉這個文件!
12. 在Import時不要使用表達(dá)式或者路徑
在系統(tǒng)的命令行里使用文件夾路徑或者文件的擴(kuò)展名,但不要在import語句中使用。即,使用import mod,而不是import mod.py,或者import dir/mod.py。
在實際情況中,這大概是初學(xué)者常犯的第二大錯誤了。因為模塊會有除了.py以為的其他的后綴(例如,.pyc),強(qiáng)制寫上某個后綴不僅是不合語法的,也沒有什么意義。
和系統(tǒng)有關(guān)的目錄路徑的格式是從你的模塊搜索路徑的設(shè)置里來的,而不是import語句。你可以在文件名里使用點(diǎn)來指向包的子目錄(例如,import dir1.dir2.mod),但是最左邊的目錄必須得通過模塊搜索路徑能夠找到,并且沒有在import中沒有其他路徑格式。
不正確的語句import mod.py被Python認(rèn)為是要記在一個包,它先加載一個模塊mod,然后試圖通過在一個叫做mod的目錄里去找到叫做py的模塊,最后可能什么也找不到而報出一系列費(fèi)解的錯誤信息。
13. 不要在Python中寫C代碼
以下是給不熟悉Python的C程序員的一些備忘貼士:
在if和while中條件測試時,不用輸入括號(例如,if (X==1):)。如果你喜歡的話,加上括號也無妨,只是在這里是完全多余的。
不要用分號來結(jié)束你的語句。從技術(shù)上講這在Python里是合法的,但是這毫無用處,除非你要把很多語句放在同一行里(例如,x=1; y=2; z=3)。
不要在while循環(huán)的條件測試中嵌入賦值語句(例如,while ((x=next() != NULL))。在Python中,需要表達(dá)式的地方不能出現(xiàn)語句,并且賦值語句不是一個表達(dá)式。
03 編程錯誤
下面終于要講到當(dāng)你用到更多的Python的功能(數(shù)據(jù)類型,函數(shù),模塊,類等等)時可能碰到的問題了。由于篇幅有限,這里盡量精簡,尤其是對一些高級的概念。要想了解更多的細(xì)節(jié),敬請閱讀《Python學(xué)習(xí)手冊》。
14. 打開文件的調(diào)用不使用模塊搜索路徑
當(dāng)你在Python中調(diào)用open()來訪問一個外部的文件時,Python不會使用模塊搜索路徑來定位這個目標(biāo)文件。它會使用你提供的絕對路徑,或者假定這個文件是在當(dāng)前工作目錄中。模塊搜索路徑僅僅為模塊加載服務(wù)的。
15. 不同的類型對應(yīng)的方法也不同
列表的方法是不能用在字符串上的,反之亦然。通常情況下,方法的調(diào)用是和數(shù)據(jù)類型有關(guān)的,但是內(nèi)部函數(shù)通常在很多類型上都可以使用。舉個例子來說,列表的reverse方法僅僅對列表有用,但是len函數(shù)對任何具有長度的對象都適用。
16. 不能直接改變不可變數(shù)據(jù)類型
記住你沒法直接的改變一個不可變的對象(例如,元組,字符串):
T = (1, 2, 3)
T[2] = 4 # 錯誤
用切片,聯(lián)接等構(gòu)建一個新的對象,并根據(jù)需求將原來變量的值賦給它。因為Python會自動回收沒有用的內(nèi)存,因此這沒有看起來那么浪費(fèi):
T = T[:2] + (4,) # 沒問題了: T 變成了 (1, 2, 4)
17. 使用簡單的for循環(huán)而不是while或者range
當(dāng)你要從左到右遍歷一個有序的對象的所有元素時,用簡單的for循環(huán)(例如,for x in seq:)相比于基于while-或者range-的計數(shù)循環(huán)而言會更容易寫,通常運(yùn)行起來也更快。
除非你一定需要,盡量避免在一個for循環(huán)里使用range:讓Python來替你解決標(biāo)號的問題。在下面的例子中三個循環(huán)結(jié)構(gòu)都沒有問題,但是第一個通常來說更好;在Python里,簡單至上。
S = "lumberjack"
for c in S: print c # 最簡單
for i in range(len(S)): print S[i] # 太多了
i = 0 # 太多了
while i < len(S): print S[i]; i += 1
12下一頁>(免責(zé)聲明:本網(wǎng)站內(nèi)容主要來自原創(chuàng)、合作伙伴供稿和第三方自媒體作者投稿,凡在本網(wǎng)站出現(xiàn)的信息,均僅供參考。本網(wǎng)站將盡力確保所提供信息的準(zhǔn)確性及可靠性,但不保證有關(guān)資料的準(zhǔn)確性及可靠性,讀者在使用前請進(jìn)一步核實,并對任何自主決定的行為負(fù)責(zé)。本網(wǎng)站對有關(guān)資料所引致的錯誤、不確或遺漏,概不負(fù)任何法律責(zé)任。
任何單位或個人認(rèn)為本網(wǎng)站中的網(wǎng)頁或鏈接內(nèi)容可能涉嫌侵犯其知識產(chǎn)權(quán)或存在不實內(nèi)容時,應(yīng)及時向本網(wǎng)站提出書面權(quán)利通知或不實情況說明,并提供身份證明、權(quán)屬證明及詳細(xì)侵權(quán)或不實情況證明。本網(wǎng)站在收到上述法律文件后,將會依法盡快聯(lián)系相關(guān)文章源頭核實,溝通刪除相關(guān)內(nèi)容或斷開相關(guān)鏈接。 )