Python異常處理
異常概念:
異常:就是不正常的情況,程序開發(fā)過程中錯(cuò)誤和BUG都是補(bǔ)充正常的情況
異常發(fā)生的后果:引發(fā)程序崩潰
處理異常(容錯(cuò)):包容出現(xiàn)的不正常的錯(cuò)誤懒棉,保證程序的正常執(zhí)行
python提供了兩個(gè)非常重要的功能來處理python程序在運(yùn)行中出現(xiàn)的異常和錯(cuò)誤。
try-except:異常捕獲處理
assertions:斷言測試
基本語法:
try:
可能出現(xiàn)異常的代碼
except
<異常名稱>:
一旦出現(xiàn)異常執(zhí)行的代碼
else:
沒有異常時(shí)執(zhí)行的代碼
finally:
不論是否有異常都會(huì)執(zhí)行的代碼
python標(biāo)準(zhǔn)異常
異常名稱 ? ? ? ? ? ? ? ? ? ? ? ? ? ?描述
BaseException ? ? ? ? ? ??所有異常的基類
SystemExit ? ? ? ? ? ? ? ? ? ?解釋器請求退出
KeyboardInterrupt ? ? ? ??用戶中斷執(zhí)行(通常是輸入^C)
Exception? ? ? ? ? ? ? ? ? ? ??常規(guī)錯(cuò)誤的基類
StopIteration? ? ? ? ? ? ? ? ?迭代器沒有更多的值
GeneratorExit? ? ? ? ? ? ? ?生成器(generator)發(fā)生異常來通知退出
StandardError? ? ? ? ? ? ??所有的內(nèi)建標(biāo)準(zhǔn)異常的基類
FloatingPointError ? ? ? ?浮點(diǎn)計(jì)算錯(cuò)誤
OverflowError ? ? ? ? ? ? ? ?數(shù)值運(yùn)算超出最大限制
ZeroDivisionError ? ? ? ??除(或取模)零(所有數(shù)據(jù)類型)
AssertionError ? ? ? ? ? ? ??斷言語句失敗
AttributeError ? ? ? ? ? ? ? ??對象沒有這個(gè)屬性
EOFError ? ? ? ? ? ? ? ? ? ? ??沒有內(nèi)建輸入,到達(dá)EOF標(biāo)記
EnvironmentError ? ? ? ? ?操作系統(tǒng)錯(cuò)誤的基類
IOError ? ? ? ? ? ? ? ? ? ? ? ? ? ?輸入/輸出操作失敗
OSError ? ? ? ? ? ? ? ? ? ? ? ? ?操作系統(tǒng)錯(cuò)誤
WindowsError ? ? ? ? ? ? ??系統(tǒng)調(diào)用失敗
ImportError ? ? ? ? ? ? ? ? ? ?導(dǎo)入模塊/對象失敗
LookupError ? ? ? ? ? ? ? ??無效數(shù)據(jù)查詢的基類
IndexError ? ? ? ? ? ? ? ? ? ?序列中沒有此索引(index)
KeyError ? ? ? ? ? ? ? ? ? ? ?映射中沒有這個(gè)鍵
MemoryError ? ? ? ? ? ? ??內(nèi)存溢出錯(cuò)誤(對于Python解釋器不是致命的)
NameError ? ? ? ? ? ? ? ? ?未聲明/初始化對象(沒有屬性)
UnboundLocalError ? ? 訪問未初始化的本地變量
ReferenceError ? ? ? ? ??弱引用(Weak reference)試圖訪問已經(jīng)垃圾回收了的對象
RuntimeError ? ? ? ? ? ? ??一般的運(yùn)行時(shí)錯(cuò)誤
NotImplementedError ? ?尚未實(shí)現(xiàn)的方法
SyntaxError ? ? ? ? ? ? ? ? ??Python語法錯(cuò)誤
IndentationError ? ? ? ? ? ??縮進(jìn)錯(cuò)誤
TabError ? ? ? ? ? ? ? ? ? ? ??Tab和空格混用
SystemError ? ? ? ? ? ? ? ?一般的解釋器系統(tǒng)錯(cuò)誤
TypeError ? ? ? ? ? ? ? ? ?對類型無效的操作
ValueError ? ? ? ? ? ? ? ?傳入無效的參數(shù)
UnicodeError ? ? ? ? ? ? ??Unicode相關(guān)的錯(cuò)誤
UnicodeDecodeError ? ? ?Unicode解碼時(shí)的錯(cuò)誤
UnicodeEncodeError ? ? ? Unicode編碼時(shí)錯(cuò)誤
UnicodeTranslateError ? ? ?Unicode轉(zhuǎn)換時(shí)錯(cuò)誤
Warning ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?警告的基類
DeprecationWarning ? ? ? ? ??關(guān)于被棄用的特征的警告
FutureWarning ? ? ? ? ? ? ? ? ? ? ? ? ??關(guān)于構(gòu)造將來語義會(huì)有改變的警告
OverflowWarning ? ? ? ? ? ? ? ? ?舊的關(guān)于自動(dòng)提升為長整型(long)的警告
PendingDeprecationWarning ? ? ? ? ? ?關(guān)于特性將會(huì)被廢棄的警告
RuntimeWarning ? ? ? ? ? ? ? ? ? ? ?可疑的運(yùn)行時(shí)行為(runtime behavior)的警告
SyntaxWarning ? ? ? ? ? ? ? ? ? ? ? ?可疑的語法的警告
UserWarning ? ? ? ? ? ? ? ? ? ? ? ? ??用戶代碼生成的警告
什么是異常屈芜?
異常即是一個(gè)事件签钩,該事件會(huì)在程序執(zhí)行過程中發(fā)生硬霍,影響了程序的正常執(zhí)行川抡。一般情況下,在Python無法正常處理程序時(shí)就會(huì)發(fā)生一個(gè)異常须尚。異常是Python對象,表示一個(gè)錯(cuò)誤侍咱。
當(dāng)Python腳本發(fā)生異常時(shí)我們需要捕獲處理它耐床,否則程序會(huì)終止執(zhí)行。
異常處理
捕捉異承ǜ可以使用try/except語句撩轰。
try/except語句用來檢測try語句塊中的錯(cuò)誤,從而讓except語句捕獲異常信息并處理昧廷。
如果你不想在異常發(fā)生時(shí)結(jié)束你的程序堪嫂,只需在try里捕獲它。
語法:
以下為簡單的try....except...else的語法:
try:
<語句>#運(yùn)行別的代碼
except<名字>:
<語句>#如果在try部份引發(fā)了'name'異常
except<名字>木柬,<數(shù)據(jù)>:
<語句>#如果引發(fā)了'name'異常皆串,獲得附加的數(shù)據(jù)
else:
<語句>#如果沒有異常發(fā)生
try的工作原理是,當(dāng)開始一個(gè)try語句后眉枕,python就在當(dāng)前程序的上下文中作標(biāo)記恶复,這樣當(dāng)異常出現(xiàn)時(shí)就可以回到這里怜森,try子句先執(zhí)行,接下來會(huì)發(fā)生什么依賴于執(zhí)行時(shí)是否出現(xiàn)異常谤牡。
·如果當(dāng)try后的語句執(zhí)行時(shí)發(fā)生異常副硅,python就跳回到try并執(zhí)行第一個(gè)匹配該異常的except子句,異常處理完畢翅萤,控制流就通過整個(gè)try語句(除非在處理異常時(shí)又引發(fā)新的異常)恐疲。
·如果在try后的語句里發(fā)生了異常,卻沒有匹配的except子句套么,異常將被遞交到上層的try培己,或者到程序的最上層(這樣將結(jié)束程序,并打印缺省的出錯(cuò)信息)违诗。
·如果在try子句執(zhí)行時(shí)沒有發(fā)生異常漱凝,python將執(zhí)行else語句后的語句(如果有else的話),然后控制流通過整個(gè)try語句诸迟。
實(shí)例
下面是簡單的例子茸炒,它打開一個(gè)文件,在該文件中的內(nèi)容寫入內(nèi)容阵苇,且并未發(fā)生異常:
#!/usr/bin/python
# -*- coding: UTF-8 -*-
try:
fh=open("testfile","w")
fh.write("這是一個(gè)測試文件壁公,用于測試異常!!")
exceptIOError:
print"Error:沒有找到文件或讀取文件失敗"
else:
print"內(nèi)容寫入文件成功"
fh.close()
以上程序輸出結(jié)果:
$ python test.py
內(nèi)容寫入文件成功
$ cat testfile#查看寫入的內(nèi)容
這是一個(gè)測試文件,用于測試異常!!
實(shí)例
下面是簡單的例子绅项,它打開一個(gè)文件紊册,在該文件中的內(nèi)容寫入內(nèi)容,但文件沒有寫入權(quán)限快耿,發(fā)生了異常:
#!/usr/bin/python
# -*- coding: UTF-8 -*-
try:
fh=open("testfile","w")
fh.write("這是一個(gè)測試文件囊陡,用于測試異常!!")
exceptIOError:
print"Error:沒有找到文件或讀取文件失敗"
else:
print"內(nèi)容寫入文件成功"
fh.close()
在執(zhí)行代碼前為了測試方便,我們可以先去掉testfile文件的寫權(quán)限掀亥,命令如下:
chmod-w testfile
再執(zhí)行以上代碼:
$ python test.py
Error:沒有找到文件或讀取文件失敗
使用except而不帶任何異常類型
你可以不帶任何異常類型使用except撞反,如下實(shí)例:
try:
正常的操作
......................
except:
發(fā)生異常,執(zhí)行這塊代碼
......................
else:
如果沒有異常執(zhí)行這塊代碼
以上方式try-except語句捕獲所有發(fā)生的異常搪花。但這不是一個(gè)很好的方式遏片,我們不能通過該程序識別出具體的異常信息。因?yàn)樗东@所有的異常撮竿。
使用except而帶多種異常類型
處理多個(gè)異常
?try:
?可能出現(xiàn)異常的代碼
?except Excp1:
?出現(xiàn)Excp1異常之后執(zhí)行的代碼
?except Excp2:
?出現(xiàn)Excp1異常之后執(zhí)行的代碼
你也可以使用相同的except語句來處理多個(gè)異常信息吮便,如下所示:
try:
正常的操作
......................
except(Exception1[,Exception2[,...ExceptionN]]]):
發(fā)生以上多個(gè)異常中的一個(gè),執(zhí)行這塊代碼
......................
else:
如果沒有異常執(zhí)行這塊代碼
try-finally語句
try-finally語句無論是否發(fā)生異常都將執(zhí)行最后的代碼幢踏。
try:
<語句>
finally:
<語句>#退出try時(shí)總會(huì)執(zhí)行
raise
實(shí)例
#!/usr/bin/python
# -*- coding: UTF-8 -*-
try:
fh=open("testfile","w")
fh.write("這是一個(gè)測試文件髓需,用于測試異常!!")
finally:
print"Error:沒有找到文件或讀取文件失敗"
如果打開的文件沒有可寫權(quán)限,輸出如下所示:
$ python test.py
Error:沒有找到文件或讀取文件失敗
同樣的例子也可以寫成如下方式:
#!/usr/bin/python
# -*- coding: UTF-8 -*-
try:
fh=open("testfile","w")
try:
fh.write("這是一個(gè)測試文件房蝉,用于測試異常!!")
finally:
print"關(guān)閉文件"
fh.close()
exceptIOError:
print"Error:沒有找到文件或讀取文件失敗"
當(dāng)在try塊中拋出一個(gè)異常授账,立即執(zhí)行finally塊代碼枯跑。
finally塊中的所有語句執(zhí)行后,異常被再次觸發(fā)白热,并執(zhí)行except塊代碼敛助。
參數(shù)的內(nèi)容不同于異常。
異常的參數(shù)
一個(gè)異澄萑罚可以帶上參數(shù)纳击,可作為輸出的異常信息參數(shù)。
你可以通過except語句來捕獲異常的參數(shù)攻臀,如下所示:
try:
正常的操作
......................
exceptExceptionType,Argument:
你可以在這輸出Argument的值...
變量接收的異常值通常包含在異常的語句中焕数。在元組的表單中變量可以接收一個(gè)或者多個(gè)值。
元組通常包含錯(cuò)誤字符串刨啸,錯(cuò)誤數(shù)字堡赔,錯(cuò)誤位置。
實(shí)例
以下為單個(gè)異常的實(shí)例:
#!/usr/bin/python
# -*- coding: UTF-8 -*-
#定義函數(shù)
deftemp_convert(var):
try:
returnint(var)
exceptValueError,Argument:
print"參數(shù)沒有包含數(shù)字\n",Argument
#調(diào)用函數(shù)
temp_convert("xyz");
以上程序執(zhí)行結(jié)果如下:
$ python test.py
參數(shù)沒有包含數(shù)字
invalid literalforint()withbase10:'xyz'
觸發(fā)異常
我們可以使用raise語句自己觸發(fā)異常
raise語法格式如下:
raise[Exception[,args[,traceback]]]
語句中Exception是異常的類型(例如设联,NameError)參數(shù)是一個(gè)異常參數(shù)值善已。該參數(shù)是可選的,如果不提供离例,異常的參數(shù)是"None"换团。
最后一個(gè)參數(shù)是可選的(在實(shí)踐中很少使用),如果存在宫蛆,是跟蹤異常對象艘包。
實(shí)例
一個(gè)異常可以是一個(gè)字符串耀盗,類或?qū)ο笙牖ⅰython的內(nèi)核提供的異常,大多數(shù)都是實(shí)例化的類叛拷,這是一個(gè)類的實(shí)例的參數(shù)磷醋。
定義一個(gè)異常非常簡單,如下所示:
deffunctionName(level):
iflevel<1:
raiseException("Invalid level!",level)
#觸發(fā)異常后胡诗,后面的代碼就不會(huì)再執(zhí)行
注意:為了能夠捕獲異常,"except"語句必須有用相同的異常來拋出類對象或者字符串淌友。
例如我們捕獲以上異常煌恢,"except"語句如下所示:
try:
正常邏輯
except"Invalid level!":
觸發(fā)自定義異常
else:
其余代碼
實(shí)例
#!/usr/bin/python
# -*- coding: UTF-8 -*-
#定義函數(shù)
defmye(level):
iflevel<1:
raiseException("Invalid level!",level)
#觸發(fā)異常后,后面的代碼就不會(huì)再執(zhí)行
try:
mye(0)//觸發(fā)異常
except"Invalid level!":
print1
else:
print2
執(zhí)行以上代碼震庭,輸出結(jié)果為:
$ python test.py
Traceback(most
recent calllast):
File"test.py",line11,in
mye(0)
File"test.py",line7,inmye
raiseException("Invalid level!",level)
Exception:('Invalid level!',0)
用戶自定義異常
通過創(chuàng)建一個(gè)新的異常類瑰抵,程序可以命名它們自己的異常。異常應(yīng)該是典型的繼承自Exception類器联,通過直接或間接的方式二汛。
以下為與RuntimeError相關(guān)的實(shí)例,實(shí)例中創(chuàng)建了一個(gè)類婿崭,基類為RuntimeError,用于在異常觸發(fā)時(shí)輸出更多的信息肴颊。
在try語句塊中氓栈,用戶自定義的異常后執(zhí)行except塊語句,變量e是用于創(chuàng)建Networkerror類的實(shí)例婿着。
classNetworkerror(RuntimeError):
def__init__(self,arg):
self.args=arg
在你定義以上類后授瘦,你可以觸發(fā)該異常,如下所示:
try:
raiseNetworkerror("Bad
hostname")
exceptNetworkerror,e:
printe.args