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