Python 是面向對象的語言,所以程序拋出的異常也是類。
常見的異常類
1.NameError:嘗試訪問一個沒有申明的變量
2.ZeroDivisionError:除數為 0
3.SyntaxError:語法錯誤
4.IndexError:索引超出序列范圍
5.KeyError:請求一個不存在的字典關鍵字
6.IOError:輸入輸出錯誤(比如你要讀的文件不存在)
7.AttributeError:嘗試訪問未知的對象屬性
8.TypeError:傳給函數的參數類型不正確,比如給 int 函數傳入字符型
自定義異常類
盡管內建的異常類已經包括大部分情況,但如果需要創建自己的異常類,可以采用下面的方法:
class CustomException(Exception): # Exception 為所有異常的基類
# 處理異常的代碼
捕獲異常
try:
# 需要進行捕獲異常的代碼,只有出現異常之前的代碼被最終執行
except (Exception1,Exception2,...) as argument:
# 捕獲到 (Exception1,Exception2,...) 里的異常才執行本段代碼
# argument 是一個異常類的實例,包含異常的具體信息
except:
# 捕獲到 (Exception1,Exception2,...) 以外的異常執行本段代碼,用 sys 模塊的 exc_info() 函數可以獲取異常信息
else:
# 如果沒有捕獲到異常則執行本段代碼
finally:
# 無論是否捕獲到異常都執行本段代碼
上述語句并不都是必須的,譬如 try...except...、try...finally... 或者 try...except...else... 語句都是可行的。我們還可以 用 try...except...else... 代替 if...else...。
拋出異常
如果我們想要在自己編寫的程序中主動拋出異常,可以采用如下兩種方法:
raise Exception(reason)
Exception 必須是一個異常類的名稱。可選項 reason 用來傳遞異常的信息。
assert expression[,reason]
assert 是斷言的關鍵字。當表達式 expression 為真則什么都不做,否則拋出 AssertionError 異常。reason 提供異常的信息。
上下文管理器
由于對象 File 支持上下文管理協議,因此可以采用下面的方法打開文件:
with open('filename') as fp:
# 無論本段代碼是否出現異常,文件對象 fp 均能正確關閉
Example
import sys
def div(num, den):
print('_________________ (',num,',',den,')\n')
try:
ans = num/den
assert den != num, 'Equal' # 斷言:分子分母不相等
den = 'Changed' # 如果執行本語句之前未出現異常,改變 den 的值
if num % 2: # 如果分子為奇數,則拋出異常
raise ValueError('Odd')
except ZeroDivisionError as e:
print('except ... as ...\n\t', e)
except:
print('except\n\t', sys.exc_info())
else:
print('else\n\t', ans)
finally:
print('finally\n\t', den)
div(1,0) # 除數為零,為 ZeroDivisionError 異常類
div(1,1) # 分子等于分母,斷言為假,拋出異常
div(2,1) # 無異常
div(3,1) # 分子為奇數,通過 raise 拋出異常
div(3,'x') # 不屬于 ZeroDivisionError 的其他異常
運行結果:
_________________ ( 1 , 0 )
except ... as ...
division by zero
finally
0
_________________ ( 1 , 1 )
except
(<class 'AssertionError'>, AssertionError('Equal',), <traceback object at 0x00000000029B42C8>)
finally
1
_________________ ( 2 , 1 )
else
2.0
finally
Changed
_________________ ( 3 , 1 )
except
(<class 'ValueError'>, ValueError('Odd',), <traceback object at 0x00000000029B42C8>)
finally
Changed
_________________ ( 3 , x )
except
(<class 'TypeError'>, TypeError("unsupported operand type(s) for /: 'int' and 'str'",), <traceback object at 0x00000000029B42C8>)
finally
x