Python基礎(chǔ)教程之異常處理詳解
前言
最近發(fā)現(xiàn)有些東西長時間不用就要忘了,堅持每天復(fù)習(xí)總結(jié)一個小知識點吧~
異常是什么呢?就是在代碼執(zhí)行過程中非預(yù)期的執(zhí)行結(jié)果,隨著代碼越來越復(fù)雜,代碼中的執(zhí)行邏輯也會越來越復(fù)雜,如果沒有處理好異常情況,很有可能造成軟件執(zhí)行錯誤,導(dǎo)致重大損失。相反,如果合理的處理異常情況,則可以增強軟件的穩(wěn)定性,提高體驗感。
異常
在Python中,使用異常對象(exception object)來表示代碼執(zhí)行過程中所發(fā)生的異常情況,當執(zhí)行程序爆出錯誤的時候則會拋出異常。
如果沒有正確處理異常,則會終止運行。
你可以想象一下,如果你在開發(fā)一款產(chǎn)品時,出現(xiàn)異常而不報告出發(fā)生異常的原因,是不是會很難受,也很難解決異常的問題。
為了提高產(chǎn)品的穩(wěn)定性與靈活性,Python運行開發(fā)者捕捉并處理各類異常,一般的內(nèi)部模塊報錯如KeyError異常類較為常見,當然也有很多其他的。
錯誤與異常
簡單了解下,Python將代碼執(zhí)行錯誤分為兩類:語法錯誤(syntax error)和異常(exception)。
語法錯誤
首先,是語法錯誤的問題。字面可知,代碼解析錯誤。
這種錯誤通常出現(xiàn)在初學(xué)者,主要原因是所執(zhí)行的代碼不符合Python語法的規(guī)范,故會報出語法錯誤導(dǎo)致code停機。
下面給出一個錯誤示范:
>>> if for not in list(1,2,3,4)
File "<stdin>", line 1
if for not in list(1,2,3,4)
^
SyntaxError: invalid syntax
從上面的報錯中可以知道是語法錯誤,無效的語法。并且指出了錯誤的地方在for的位置,很顯然if和for不能用在一起。
異常
異常則是在代碼執(zhí)行過程中發(fā)現(xiàn)的錯誤,這是很難提前被發(fā)現(xiàn)的,即使代碼寫的很規(guī)范標準,但也可能會出現(xiàn)執(zhí)行異常的情況。
下面給出一個錯誤示例。經(jīng)常被提到的是除數(shù)為0的情況,數(shù)學(xué)運算中,0是不可以作為除數(shù)的,如果作為除數(shù)則會爆出異常。
>>> a = 5 / 0 Traceback (most recent call last): File "<stdin>", line 1, in <module> ZeroDivisionError: division by zero
很明顯報錯指出0作為除數(shù)是不對的。
異常處理
當我們了解了錯誤和異常后,就得處理異常情況。對于語法錯誤這種問題多多練習(xí)即可避免,對于第二種異常,python給出了try-except語句來處理。
為了更清晰的解釋,還是以上面的除數(shù)為0的情況作為例子,我們通過添加try-except語句捕捉并處理異常情況:
def division(x, y):
try:
return x / y
except ZeroDivisionError:
print("0不能作為除數(shù)!??!")
然后我們再在terminal調(diào)用函數(shù)試試:
division(x=1, y=0)
輸出:
0不能作為除數(shù)!?。?/p>
雖然我們的輸入是錯誤的理論應(yīng)該報錯,但是并沒有報錯而且還返回了一段話。
下面解釋一下try-except語句的工作原理:
- Try-except中的代碼會被正常執(zhí)行
- 如果沒有出現(xiàn)異常則跳過except代碼塊并結(jié)束try-except
- 如果try-except中的某一句代碼出現(xiàn)了問題異常,剩余代碼停止執(zhí)行,如果出現(xiàn)的異常與except所指定的一致,則執(zhí)行except中的代碼塊,異常處理結(jié)束整個應(yīng)用程序繼續(xù)執(zhí)行
- 如果出現(xiàn)的異常與except中指定的不符合,那么則跳出try語句,程序繼續(xù)拋出異常并終止執(zhí)行代碼
當然,我們寫的代碼可能不止出現(xiàn)一種異常情況,我們可以將所有異常寫在一個except語句中,如下所示:
except (RuntimeError, TypeError, NameError):pass
如果捕捉到異常列表中的任意一項異常表達式,則都會進入except處理。
當然,如果你想對每一種異常進行單獨處理也可以一個一個的分開進行處理:
def passpass(x=1):
try:
return print(x+x)
except (RuntimeError, TypeError, NameError):
pass
except TypeError:
print('參數(shù)錯誤啦~')
except NameError:
print('名稱錯誤啦~')
今天先到這里明兒再更,去跳繩了,哎,每天卷卷卷,老北京雞內(nèi)卷啊~(2022.4.20)
Python的異常類型是可以繼承的, 我們在此僅需知道,若except后指定的異常繼承自前面異常,由此后面的異常也會被捕捉到,示例如下:
class A(Exception):
pass
class B(A):
pass
class C(B):
pass
for cls in [A, B, C]:
try:
raise cls()
except C:
print('C')
except B:
print('B')
except A:
print('A')
則會輸出:
A
B
C
相反,如果把except的順序倒過來,則只會輸出A,由于異常B和C都繼承來自A,由此在捕捉到B異常后則會終止:
class A(Exception):
pass
class B(A):
pass
class C(B):
pass
for cls in [A, B, C]:
try:
raise cls()
except A:
print('A')
except B:
print('B')
except C:
print('C')
輸出全為A:
A
A
A
如果所有異常都一一的提取出來,這會很麻煩,此時可以在最后的一個except中不設(shè)置異常類型,由此剩下沒有被捕獲的異常全會被捕捉:
def passpass(x=1):
try:
return print(x/x)
except TypeError:
print('參數(shù)錯誤啦~')
except NameError:
print('名稱錯誤啦~')
except:
print('報錯啦~自己查')
一個重要功能:else.
else語句則是用來執(zhí)行一些額外操作,如try代碼塊中執(zhí)行了一些文件操作,在else中可以釋放資源,else的語法格式如下:
try: pass except: pass else: pass
當然我們還可以操作異常,如下:
def passpass(x=1):
try:
return print(x/x)
except TypeError as error:
print('參數(shù)錯誤啦~', error)
except NameError:
print('名稱錯誤啦~')
except:
print('報錯啦~自己查')
自主拋出異常
開發(fā)者在平常的有些情況下并沒有執(zhí)行錯誤,但是不符合設(shè)計邏輯,由此需要開發(fā)者主動拋出異常,這時我們需要使用raise語句拋出異常:
>>> raise NameError('HiThere')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
NameError: HiThere
自定義異常
Python內(nèi)置異常無法滿足開發(fā)需求的時候,可以自定義異常。
自定義異常類必須要直接或間接繼承自Exception類。自定義異常類可以像其他類一樣做任何事情,但原則上要保持簡潔,提高一些屬性即可。
class Error(Exception): """Base class for exception in this module.""" pass class InputError(Error): """Exception raised for errors in the input. Attributes: expression -- input expression in which the error occurred message -- explanation of the winerror """ def __init__(self, expression, message): self.expression = expression self.message = message
finally子句
else在代碼正常執(zhí)行后才會被執(zhí)行的代碼塊,但有些情況無論代碼塊是否出現(xiàn)異常都要執(zhí)行,則需要用到finally語句:
def passpass(x=1):
try:
return print(x/x)
except TypeError as error:
print('參數(shù)錯誤啦~', error)
except NameError:
print('名稱錯誤啦~')
except:
print('報錯啦~自己查')
finally:
print('運算結(jié)束~')
總結(jié)
到此這篇關(guān)于Python基礎(chǔ)教程之異常處理的文章就介紹到這了,更多相關(guān)Python異常處理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python?async+request與async+aiohttp實現(xiàn)異步網(wǎng)絡(luò)請求探索
這篇文章主要介紹了Python?async+request與async+aiohttp實現(xiàn)異步網(wǎng)絡(luò)請求探索,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2022-10-10
Python Pymysql實現(xiàn)數(shù)據(jù)存儲的示例
本文主要介紹了Python Pymysql實現(xiàn)數(shù)據(jù)存儲的示例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03
解決在pycharm運行代碼,調(diào)用CMD窗口的命令運行顯示亂碼問題
今天小編就為大家分享一篇解決在pycharm運行代碼,調(diào)用CMD窗口的命令運行顯示亂碼問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-08-08
python爬蟲實現(xiàn)POST request payload形式的請求
這篇文章主要介紹了python爬蟲實現(xiàn)POST request payload形式的請求,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-04-04

