python3 異常處理

1.異常和錯誤

1.1 語法錯誤

#語法錯誤示範一
if

#語法錯誤示範二
def test: pass

#語法錯誤示範三
print(haha

1.2 邏輯錯誤

#用戶輸入不完整(好比輸入爲空)或者輸入非法(輸入不是數字)
num=input(">>: ") int(num) #沒法完成計算
res1=1/0 res2=1+'str'

2.python中異常種類

AttributeError 試圖訪問一個對象沒有的屬性,好比foo.x,可是foo沒有屬性x IOError 輸入/輸出異常;基本上是沒法打開文件 ImportError 沒法引入模塊或包;基本上是路徑問題或名稱錯誤 IndentationError 語法錯誤(的子類) ;代碼沒有正確對齊 IndexError 下標索引超出序列邊界,好比當x只有三個元素,卻試圖訪問x[5] KeyError 試圖訪問字典裏不存在的鍵 KeyboardInterrupt Ctrl+C被按下 NameError 嘗試訪問一個沒有申明的變量 SyntaxError Python代碼非法,代碼不能編譯(我的認爲這是語法錯誤,寫錯了) TypeError 傳入對象類型與要求的不符合 UnboundLocalError 試圖訪問一個還未被設置的局部變量,基本上是因爲另有一個同名的全局變量,致使你覺得正在訪問它 ValueError 傳入一個調用者不指望的值,即便值的類型是正確的

補充python

異常名稱 描述 BaseException 全部異常的基類 SystemExit 解釋器請求退出 KeyboardInterrupt 用戶中斷執行(一般是輸入^C) Exception 常規錯誤的基類 StopIteration 迭代器沒有更多的值 GeneratorExit 生成器(generator)發生異常來通知退出 StandardError 全部的內建標準異常的基類 ArithmeticError 全部數值計算錯誤的基類 FloatingPointError 浮點計算錯誤 OverflowError 數值運算超出最大限制 ZeroDivisionError 除(或取模)零 (全部數據類型) AssertionError 斷言語句失敗 AttributeError 對象沒有這個屬性 EOFError 沒有內建輸入,到達EOF 標記 EnvironmentError 操做系統錯誤的基類 IOError 輸入/輸出操做失敗 OSError 操做系統錯誤 WindowsError 系統調用失敗 ImportError 導入模塊/對象失敗 LookupError 無效數據查詢的基類 IndexError 序列中沒有此索引(index) KeyError 映射中沒有這個鍵 MemoryError 內存溢出錯誤(對於Python 解釋器不是致命的) NameError 未聲明/初始化對象 (沒有屬性) UnboundLocalError 訪問未初始化的本地變量 ReferenceError 弱引用(Weak reference)試圖訪問已經垃圾回收了的對象 RuntimeError 通常的運行時錯誤 NotImplementedError 還沒有實現的方法 SyntaxError Python 語法錯誤 IndentationError 縮進錯誤 TabError Tab 和空格混用 SystemError 通常的解釋器系統錯誤 TypeError 對類型無效的操做 ValueError 傳入無效的參數 UnicodeError Unicode 相關的錯誤 UnicodeDecodeError Unicode 解碼時的錯誤 UnicodeEncodeError Unicode 編碼時錯誤 UnicodeTranslateError Unicode 轉換時錯誤 Warning 警告的基類 DeprecationWarning 關於被棄用的特徵的警告 FutureWarning 關於構造未來語義會有改變的警告 OverflowWarning 舊的關於自動提高爲長整型(long)的警告 PendingDeprecationWarning 關於特性將會被廢棄的警告 RuntimeWarning 可疑的運行時行爲(runtime behavior)的警告 SyntaxWarning 可疑的語法的警告 UserWarning 用戶代碼生成的警告

3.異常處理

3.1 什麼是異常?

  異常發生以後,異常以後的代碼就不會執行了。git

3.2 什麼是異常處理

  python解釋器檢測到錯誤,觸發異常(也容許程序員本身觸發異常)程序員

程序員編寫特定的代碼,專門用來捕捉這個異常(這段代碼與程序邏輯無關,與異常處理有關)安全

若是捕捉成功則進入另一個處理分支,執行你爲其定製的邏輯,使程序不會崩潰,這就是異常處理編碼

3.3 爲何要進行異常處理

  python解析器去執行程序,檢測到了一個錯誤時,觸發異常,異常觸發後且沒被處理的狀況下,程序就在當前異常處終止,後面的代碼不會運行,誰會去用一個運行着忽然就崩潰的軟件。spa

因此你必須提供一種異常處理機制來加強你程序的健壯性與容錯性 操作系統

3.4 如何處理異常

(1)使用if判斷式code

#_*_coding:utf-8_*_
__author__ = 'Linhaifeng' num1=input('>>: ') #輸入一個字符串試試
if num1.isdigit(): int(num1) #咱們的正統程序放到了這裏,其他的都屬於異常處理範疇
elif num1.isspace(): print('輸入的是空格,就執行我這裏的邏輯') elif len(num1) == 0: print('輸入的是空,就執行我這裏的邏輯') else: print('其餘情狀況,執行我這裏的邏輯') ''' 問題一: 使用if的方式咱們只爲第一段代碼加上了異常處理,但這些if,跟你的代碼邏輯並沒有關係,這樣你的代碼會由於可讀性差而不容易被看懂 問題二: 這只是咱們代碼中的一個小邏輯,若是相似的邏輯多,那麼每一次都須要判斷這些內容,就會倒置咱們的代碼特別冗長。

總結:對象

1.if判斷式的異常處理只能針對某一段代碼,對於不一樣的代碼段的相同類型的錯誤你須要寫重複的if來進行處理。blog

2.在你的程序中頻繁的寫與程序自己無關,與異常處理有關的if,會使得你的代碼可讀性極其的差

3.if是能夠解決異常的,只是存在1,2的問題,因此,千萬不要妄下定論if不能用來異常處理。

def test(): print('test running') choice_dic={ '1':test } while True: choice=input('>>: ').strip() if not choice or choice not in choice_dic:continue #這即是一種異常處理機制啊
    choice_dic[choice]()

(2)python爲每一種異常定製了一個類型,而後提供了一種特定的語法結構用來進行異常處理

#基本語法
try: 被檢測的代碼塊 except 異常類型: try中一旦檢測到異常,就執行這個位置的邏輯
#未添加異常處理的代碼
f = open('a.txt') g = (line.strip() for line in f) for line in g: print(line) else: f.close()
#添加異常處理的代碼
try: f = open('a.txt') g = (line.strip() for line in f) print(next(g)) print(next(g)) print(next(g)) print(next(g)) print(next(g)) except StopIteration: f.close() ''' next(g)會觸發迭代f,依次next(g)就能夠讀取文件的一行行內容,不管文件a.txt有多大,同一時刻內存中只有一行內容。 提示:g是基於文件句柄f而存在的,於是只能在next(g)拋出異常StopIteration後才能夠執行f.close() '''
#異常類只能用來處理指定的異常狀況,若是非指定異常則沒法處理。 # 未捕獲到異常,程序直接報錯
 s1 = 'hello'
try: int(s1) except IndexError as e: print e
#多分支
s1 = 'hello'
try: int(s1) except IndexError as e: print(e) except KeyError as e: print(e) except ValueError as e: print(e)
#萬能異常 在python的異常中,有一個萬能異常:Exception,他能夠捕獲任意異常,即:
s1 = 'hello'
try: int(s1) except Exception as e: print(e)

你可能會說既然有萬能異常,那麼我直接用上面的這種形式就行了,其餘異常能夠忽略

你說的沒錯,可是應該分兩種狀況去看。

若是你想要的效果是,不管出現什麼異常,咱們統一丟棄,或者使用同一段代碼邏輯去處理他們,那麼騷年,大膽的去作吧,只有一個Exception就足夠了。

s1 = 'hello'
try: int(s1) except Exception,e: '丟棄或者執行其餘邏輯'
    print(e) #若是你統一用Exception,沒錯,是能夠捕捉全部異常,但意味着你在處理全部異常時都使用同一個邏輯去處理(這裏說的邏輯即當前expect下面跟的代碼塊)

 

#異常的其餘機構
s1 = 'hello'
try: int(s1) except IndexError as e: print(e) except KeyError as e: print(e) except ValueError as e: print(e) #except Exception as e: # print(e)
else: print('try內代碼塊沒有異常則執行我') finally: print('不管異常與否,都會執行該模塊,一般是進行清理工做')
#主動觸發異常
try: raise TypeError('類型錯誤') except Exception as e: print(e)
#自定義異常
class EvaException(BaseException): def __init__(self,msg): self.msg=msg def __str__(self): return self.msg try: raise EvaException('類型錯誤') except EvaException as e: print(e)

(3)try..except的方式比較if的方式的好處

  try..except這種異常處理機制就是取代if那種方式,讓你的程序在不犧牲可讀性的前提下加強健壯性和容錯性

異常處理中爲每個異常定製了異常類型(python中統一了類與類型,類型即類),對於同一種異常,一個except就能夠捕捉到,能夠同時處理多段代碼的異常(無需‘寫多個if判斷式’)減小了代碼,加強了可讀性 

 

使用try..except的方式

1:把錯誤處理和真正的工做分開來
2:代碼更易組織,更清晰,複雜的工做任務更容易實現;
3:毫無疑問,更安全了,不至於因爲一些小的疏忽而使程序意外崩潰了;

(4)何時用異常處理

  try...except應該儘可能少用,由於它自己就是你附加給你的程序的一種異常處理的邏輯,與你的主要的工做是沒有關係的,
這種東西加的多了,會致使你的代碼可讀性變差,只有在有些異常沒法預知的狀況下,才應該加上try...except,其餘的邏輯錯誤應該儘可能修正。

相關文章
相關標籤/搜索