Python和很多其他語(yǔ)言一樣罗晕,支持異常處理。我們可以使用try-catch
類似的形式捕獲異常憔儿,處理異常忆植,或者拋出異常。Python的異常命名慣例和Java語(yǔ)言有些不同谒臼, Java的異常一般以Exception
結(jié)尾朝刊,而Python的異常一般以Error
結(jié)尾。
常見(jiàn)異常
首先我們來(lái)看看常見(jiàn)異常蜈缤,這些異呈懊ィ可能由于編碼錯(cuò)誤或者其他原因?qū)е隆N覀兇蜷_(kāi)Python解釋器底哥,然后瞎打一通代碼咙鞍,應(yīng)該就能看到不少異常了。
SyntaxError
系統(tǒng)錯(cuò)誤最常見(jiàn)的原因就是編碼的縮進(jìn)錯(cuò)誤趾徽、或者缺少了分號(hào)续滋、冒號(hào)等分隔符。例如下面的例子附较。
if 0>1
print()
File "<input>", line 1
if 0>1
^
SyntaxError: invalid syntax
NameError
這個(gè)異常通常是由于使用了未定義的名稱而引起的吃粒。
fuck
Traceback (most recent call last):
File "<input>", line 1, in <module>
NameError: name 'fuck' is not defined
ZeroDivisionError
這個(gè)異常是整數(shù)除零錯(cuò)誤,一般在數(shù)學(xué)計(jì)算的時(shí)候才會(huì)出現(xiàn)拒课。
1/0
Traceback (most recent call last):
File "<input>", line 1, in <module>
ZeroDivisionError: division by zero
ValueError
ValueError
異常一般在類型轉(zhuǎn)換失敗的時(shí)候出現(xiàn)徐勃。
int('3fuck')
Traceback (most recent call last):
File "<input>", line 1, in <module>
ValueError: invalid literal for int() with base 10: '3fuck'
處理異常
捕獲異常
Java等語(yǔ)言的使用try-catch
結(jié)構(gòu)捕獲異常,在Python中也是類似的早像,不過(guò)捕獲異常使用except
關(guān)鍵字僻肖。在下面的例子中如果把第一行的fuck
注釋掉,就可以看到拋出了NameError
異常并由except
子句捕獲了卢鹦。
fuck = 'fuck!!!'
try:
fuck
except NameError:
print('This is a NameError')
多個(gè)相似的異惩卧啵可以使用同一個(gè)子句捕獲,這需要在except
子句中用括號(hào)包含多個(gè)異常類型冀自。
try:
fuck
except (NameError, ValueError):
print('This is a NameError')
如果需要對(duì)不同的異常使用不同的異常子句捕獲揉稚,則可以列出多個(gè)異常子句。當(dāng)拋出的異常是當(dāng)前異常子句中異常的實(shí)例或者子類的實(shí)例時(shí)熬粗,都會(huì)匹配到當(dāng)前子句搀玖。需要注意由于所有異常的父類是Excpetion
,所以Excpetion
類型的異常必須放到最后一個(gè)驻呐,避免其他異常子句無(wú)法執(zhí)行灌诅。
try:
fuck
except (NameError, ValueError):
print('This is a NameError')
except SyntaxError:
print('This is a SystemError')
except Exception:
print('This is a RuntimeError')
最后一個(gè)異常子句的異常類型也可以省略芳来,這樣就會(huì)捕獲所有其他未被捕獲的異常。
try:
fuck
except (NameError, ValueError):
print('This is a NameError')
except SyntaxError:
print('This is a SystemError')
except:
print('This is a RuntimeError')
如果需要獲取異常信息猜拾,可以使用as
關(guān)鍵字聲明異常變量即舌,然后就可以在異常子句中使用了。
try:
fuck
except (NameError, ValueError) as ex:
print(f'This is a NameError:{ex}')
清理資源
如果異常處理語(yǔ)句中包含了系統(tǒng)資源(文件挎袜、網(wǎng)絡(luò)連接顽聂、數(shù)據(jù)庫(kù)連接等),我們有義務(wù)在使用完畢后及時(shí)釋放這些資源盯仪。Python也提供了相應(yīng)的機(jī)制芜飘。
釋放資源主要通過(guò)兩個(gè)子句來(lái)實(shí)現(xiàn)。第一個(gè)是else
子句磨总,該子句僅當(dāng)沒(méi)有拋出異常的時(shí)候才執(zhí)行。如果有異常笼沥,這個(gè)字句就不會(huì)被執(zhí)行蚪燕。第二個(gè)子句是finally
子句,不管有沒(méi)有異常該語(yǔ)句都會(huì)執(zhí)行奔浅。利用這兩個(gè)語(yǔ)句馆纳,我們就可以優(yōu)雅的關(guān)閉資源了。
try:
raise MyError()
except (NameError, ValueError) as ex:
print(f'This is a NameError:{ex}')
except MyError:
print('This is MyError')
else:
print('Else clause')
finally:
print('This is finally clause')
拋出異常
我們可以在合適的時(shí)候拋出異常汹桦,讓上級(jí)調(diào)用者決定如何處理異常鲁驶。
下面的例子拋出了一個(gè)自定義異常。自定義異常是繼承了Exception
的類舞骆。定義之后使用raise
語(yǔ)句拋出異常钥弯。
class MyError(Exception):
pass
try:
raise MyError()
except (NameError, ValueError) as ex:
print(f'This is a NameError:{ex}')
except MyError:
print('This is MyError')
如果在except
子句中無(wú)法處理異常,需要再次向上級(jí)拋出督禽,直接使用raise
關(guān)鍵字即可脆霎。
try:
raise MyError()
except (NameError, ValueError) as ex:
print(f'This is a NameError:{ex}')
except MyError:
print('This is MyError')
raise