python至少有2類不同的錯誤:語法錯誤(SyntaxErrors)和異常(Exceptions)。
1語法錯誤
這個單詞應(yīng)該還是很有必要認(rèn)識的,呵呵,語法錯誤,也叫解析錯誤,是我們最不愿意發(fā)生的錯誤,直接拿官網(wǎng)的例子:
>>>whileTrueprint'Helloworld'
File"",line1,in?
whileTrueprint'Helloworld'
^
SyntaxError:invalidsyntax
語法錯誤提示時會先打印出現(xiàn)語法的語句然后在這語句中打上‘^’表示離語法錯誤最近的地方。例子中就是在print前少了引號(這是一個死循環(huán)~~):
>>>whileTrue:print'Helloworld'
...
Helloworld
2異常
一個語句或者一個表達(dá)式即使編譯時是沒有語法錯誤的,但是也有可能在執(zhí)行時出現(xiàn)問題,這種問題也叫異常(非致命性),異常通常都是有在程序中進(jìn)行處理的。異常是有不同類型的,常見的異常類型有ZeroDivisionError,NameErrorandTypeError,這類異常稱為標(biāo)準(zhǔn)異常,是在build-in里面定義的,可以查看Built-inExceptions。還有一類異常是用戶自定義的。
>>>10*(1/0)
Traceback(mostrecentcalllast):
File"",line1,in?
ZeroDivisionError:integerdivisionormodulobyzero>>>'2'+2
Traceback(mostrecentcalllast):
File"",line1,in?
TypeError:cannotconcatenate'str'and'int'objects
3處理異常
直接給一個比較全的異常處理的例子:打開一個txt文檔,讀入第一行的數(shù)據(jù),轉(zhuǎn)換成int數(shù)據(jù)類型,如果都成功,就打印txt總共有多少行,最后關(guān)閉文檔。
try:
f=open('test.txt','r+')
s=f.readline()
i=int(s.strip())
exceptIOErrorase:
print'I/Oerror({0}):{1}'.format(e.errno,e.strerror)
exceptValueError:
print"couldnotconvertdatatointeger"
except:
print"unexpectederror:",sys.exc_info()[0]
else:
print'therehas{0}linesinthefile'.format(len(f.readlines()))
finally:
print'endofthefunction'
f.close()
try語句處理異常,是這樣做的:
A.首選,try子語句(try和except關(guān)鍵字之間的語句)會被執(zhí)行。
B.如果沒有異常發(fā)生,except子句被略過。
C.如果有異常發(fā)生,try后面的其他語句就被跳過了,如果異常類型在except關(guān)鍵字后匹配,這個except子句被執(zhí)行。
D.如果沒有異常發(fā)生,else子句就會被執(zhí)行。else的作用是它避免了捕獲未保護的代碼所發(fā)起的異常。
E.finally子語會在try子句執(zhí)行完畢之前執(zhí)行,不管是否發(fā)生或者不發(fā)生異常。當(dāng)一個異常發(fā)生在try子句中卻未被處理時(或者發(fā)生在except或者else子句中時),finally子句執(zhí)行完后會再次拋出異常。
這些基本的語法,應(yīng)該也基本都是比較清楚的,文檔里列出了一些需要注意的地方:
第一:一次性處理多個異常時,多個異常需要用括號括起來。
except(RuntimeError,TypeError):這樣是正確的;exceptRuntimeError,TypeError:寫法是錯誤的,因為exceptValueError,e在語法上等價于exceptValueErrorase。
第二:最后一個except子句可以不帶異常類型名,這樣就可以捕獲任何未被定義的異常。
第三:當(dāng)一個異常發(fā)生時,可能它還有一些異常的參數(shù)。except語句的異常名字后面可以跟一個參數(shù),這個參數(shù)會跟異常實例綁定,存儲在instance.args中,如果異常中__str__()定義過了,就可以直接打印出參數(shù)了。
>>>try:
...raiseException('spam','eggs')
...exceptExceptionasinst:
...printtype(inst)#theexceptioninstance
...printinst.args#argumentsstoredin.args
...printinst#__str__allowsargstobeprinteddirectly
...x,y=inst.args
...print'x=',x
...print'y=',y
...
('spam','eggs')
('spam','eggs')
x=spam
y=eggs
4用戶自定義異常
用戶自定義的異常需要繼承Exception類,官網(wǎng)例子如下:
>>>classMyError(Exception):
...def__init__(self,value):
...self.value=value
...def__str__(self):
...returnrepr(self.value)
...
>>>try:
...raiseMyError(2*2)
...exceptMyErrorase:
...print'Myexceptionoccurred,value:',e.value
...
Myexceptionoccurred,value:4
>>>raiseMyError('oops!')
Traceback(mostrecentcalllast):
File"",line1,in?
__main__.MyError:'oops!'
在這個例子中,init方法被重寫了,用于創(chuàng)建一個新的成員變量value。
5已定義好的清理行為
當(dāng)不再需要這個對象的時候,有一些對象已經(jīng)定義好了標(biāo)準(zhǔn)的清理行為,不管使用這個對象操作成功或者失敗;常見的例子還是打開文檔:
forlineinopen("myfile.txt"):
printline,
這段代碼的問題是在這段代碼執(zhí)行后,文檔處于open的狀態(tài)時間是不確定的,在一個小的腳本里,這不會是一個很嚴(yán)重的問題,但是如果是一個大應(yīng)用程序中的一部分,這個問題就會被放大。使用with語句,就允許一些像files的類在使用完后能被清理完(釋放某些資源吧,我是這樣理解的):
withopen("myfile.txt")asf:
forlineinf:
printline,
換成這行代碼后,f已經(jīng)處于close狀態(tài)了。即使在讀文件里的每一行遇到錯誤,也會關(guān)閉掉。
以上內(nèi)容為大家介紹了python的錯誤和異常,希望對大家有所幫助,如果想要了解更多Python相關(guān)知識,請關(guān)注IT培訓(xùn)機構(gòu):千鋒教育。