1. 程式人生 > >python之異常處理,斷言

python之異常處理,斷言

一,異常和錯誤

part1:程式中難免出現錯誤,而錯誤分成兩種

1.語法錯誤(這種錯誤,根本過不了python直譯器的語法檢測,必須在程式執行前就改正)

#語法錯誤示範一
if

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

#語法錯誤示範三
print(haha

語法錯誤
語法錯誤

2.邏輯錯誤(邏輯錯誤)

#使用者輸入不完整(比如輸入為空)或者輸入非法(輸入不是數字)
num=input(">>: ")
int(num)

#無法完成計算
res1=1/0
res2
=1+'str'
邏輯錯誤

part2:什麼是異常

異常就是程式執行時發生錯誤的訊號,在python中,錯誤觸發的異常如下

part3:python中的異常種類

在python中不同的異常可以用不同的型別(python中統一了類與型別,型別即類)去標識,不同的類物件標識不同的異常,一個異常標識一種錯誤

# 觸發IndexError
l=['egon','aa']
l[3]
# 觸發KeyError
dic={'name':'egon'}
dic['age']

#觸發ValueError
s='hello
' int(s)
錯誤舉例
AttributeError 試圖訪問一個物件沒有的樹形,比如foo.x,但是foo沒有屬性x
IOError 輸入/輸出異常;基本上是無法開啟檔案
ImportError 無法引入模組或包;基本上是路徑問題或名稱錯誤
IndentationError 語法錯誤(的子類) ;程式碼沒有正確對齊
IndexError 下標索引超出序列邊界,比如當x只有三個元素,卻試圖訪問x[5]
KeyError 試圖訪問字典裡不存在的鍵
KeyboardInterrupt Ctrl+C被按下
NameError 使用一個還未被賦予物件的變數
SyntaxError Python程式碼非法,程式碼不能編譯(個人認為這是語法錯誤,寫錯了)
TypeError 傳入物件型別與要求的不符合
UnboundLocalError 試圖訪問一個還未被設定的區域性變數,基本上是由於另有一個同名的全域性變數,
導致你以為正在訪問它
ValueError 傳入一個呼叫者不期望的值,即使值的型別是正確的
常用異常
ArithmeticError
AssertionError
AttributeError
BaseException
BufferError
BytesWarning
DeprecationWarning
EnvironmentError
EOFError
Exception
FloatingPointError
FutureWarning
GeneratorExit
ImportError
ImportWarning
IndentationError
IndexError
IOError
KeyboardInterrupt
KeyError
LookupError
MemoryError
NameError
NotImplementedError
OSError
OverflowError
PendingDeprecationWarning
ReferenceError
RuntimeError
RuntimeWarning
StandardError
StopIteration
SyntaxError
SyntaxWarning
SystemError
SystemExit
TabError
TypeError
UnboundLocalError
UnicodeDecodeError
UnicodeEncodeError
UnicodeError
UnicodeTranslateError
UnicodeWarning
UserWarning
ValueError
Warning
ZeroDivisionError
更多異常

二,異常處理

什麼是異常?

異常發生之後

異常之後的程式碼就不執行了

什麼是異常處理

python直譯器檢測到錯誤,觸發異常(也允許程式設計師自己觸發異常)

程式設計師編寫特定的程式碼,專門用來捕捉這個異常(這段程式碼與程式邏輯無關,與異常處理有關)

如果捕捉成功則進入另外一個處理分支,執行你為其定製的邏輯,使程式不會崩潰,這就是異常處理

為什麼要進行異常處理?

python解析器去執行程式,檢測到了一個錯誤時,觸發異常,異常觸發後且沒被處理的情況下,程式就在當前異常處終止,後面的程式碼不會執行,誰會去用一個執行著突然就崩潰的軟體。

所以你必須提供一種異常處理機制來增強你程式的健壯性與容錯性 

如何進行異常處理?

首先須知,異常是由程式的錯誤引起的,語法上的錯誤跟異常處理無關,必須在程式執行前就修正

一: 使用if判斷式

num1=input('>>: ') #輸入一個字串試試
int(num1)
正常程式碼
num1=input('>>: ') #輸入一個字串試試
if num1.isdigit():
    int(num1) #我們的正統程式放到了這裡,其餘的都屬於異常處理範疇
elif num1.isspace():
    print('輸入的是空格,就執行我這裡的邏輯')
elif len(num1) == 0:
    print('輸入的是空,就執行我這裡的邏輯')
else:
    print('其他情情況,執行我這裡的邏輯')

'''
問題一:
使用if的方式我們只為第一段程式碼加上了異常處理,但這些if,跟你的程式碼邏輯並無關係,這樣你的程式碼會因為可讀性差而不容易被看懂

問題二:
這只是我們程式碼中的一個小邏輯,如果類似的邏輯多,那麼每一次都需要判斷這些內容,就會倒置我們的程式碼特別冗長。
'''
使用if判斷進行異常處理

總結:

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

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]()
你之前用的異常處理機制

二:python異常處理的'私人定製'

python:為每一種異常定製了一個型別,然後提供了一種特定的語法結構用來進行異常處理

part1:基本語法

try:
     被檢測的程式碼塊
except 異常型別:
     try中一旦檢測到異常,就執行這個位置的邏輯
f = open('a.txt')

g = (line.strip() for line in f)
for line in g:
    print(line)
else:
    f.close()
讀檔案例1
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()
'''
讀檔案例2

part2:異常類只能用來處理指定的異常情況.

複製程式碼 複製程式碼
# 未捕獲到異常,程式直接報錯
 
s1 = 'hello'
try:
    int(s1)
except IndexError as e:
    print e
複製程式碼 複製程式碼

part3:多分支

複製程式碼 複製程式碼
s1 = 'hello'
try:
    int(s1)
except IndexError as e:
    print(e)
except KeyError as e:
    print(e)
except ValueError as e:
    print(e)
複製程式碼 複製程式碼

part4:萬能異常:Exception.

他可以捕獲任意異常,即:

s1 = 'hello'
try:
    int(s1)
except Exception as e:
    print(e)

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

你說的沒錯,但是應該分兩種情況去看

1.如果你想要的效果是,無論出現什麼異常,我們統一丟棄,或者使用同一段程式碼邏輯去處理他們,那麼騷年,大膽的去做吧,只有一個Exception就足夠了。

s1 = 'hello'
try:
    int(s1)
except Exception,e:
    '丟棄或者執行其他邏輯'
    print(e)

#如果你統一用Exception,沒錯,是可以捕捉所有異常,但意味著你在處理所有異常時都使用同一個邏輯去處理(這裡說的邏輯即當前expect下面跟的程式碼塊)
Exception

2.如果你想要的效果是,對於不同的異常我們需要定製不同的處理邏輯,那就需要用到多分支了。

s1 = 'hello'
try:
    int(s1)
except IndexError as e:
    print(e)
except KeyError as e:
    print(e)
except ValueError 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)
except Exception as e:
    print(e)
多分支+Exception

part5:異常的其他機構

複製程式碼 複製程式碼
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('無論異常與否,都會執行該模組,通常是進行清理工作')
複製程式碼 複製程式碼

part6:主動觸發異常

try:
    raise TypeError('型別錯誤')
except Exception as e:
    print(e)

part7:自定義異常

複製程式碼 複製程式碼
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)
複製程式碼 複製程式碼

part8:斷言

# assert 條件
 
assert 1 == 1
 
assert 1 == 2

part9:try..except的方式比較if的方式的好處

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

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

使用try..except的方式

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

什麼時候用異常處理

有的同學會這麼想,學完了異常處理後,好強大,我要為我的每一段程式都加上try...except,幹毛線去思考它會不會有邏輯錯誤啊,這樣就很好啊,多省腦細胞===》2B青年歡樂多

try...except應該儘量少用,因為它本身就是你附加給你的程式的一種異常處理的邏輯,與你的主要的工作是沒有關係的
這種東西加的多了,會導致你的程式碼可讀性變差,只有在有些異常無法預知的情況下,才應該加上try...except,其他的邏輯錯誤應該儘量修正

一,異常和錯誤

part1:程式中難免出現錯誤,而錯誤分成兩種

1.語法錯誤(這種錯誤,根本過不了python直譯器的語法檢測,必須在程式執行前就改正)

#語法錯誤示範一
if

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

#語法錯誤示範三
print(haha

語法錯誤
語法錯誤

2.邏輯錯誤(邏輯錯誤)

#使用者輸入不完整(比如輸入為空)或者輸入非法(輸入不是數字)
num=input(">>: ")
int(num)

#無法完成計算
res1=1/0
res2=1+'str'
邏輯錯誤

part2:什麼是異常

異常就是程式執行時發生錯誤的訊號,在python中,錯誤觸發的異常如下

part3:python中的異常種類

在python中不同的異常可以用不同的型別(python中統一了類與型別,型別即類)去標識,不同的類物件標識不同的異常,一個異常標識一種錯誤

# 觸發IndexError
l=['egon','aa']
l[3]
# 觸發KeyError
dic={'name':'egon'}
dic['age']

#觸發ValueError
s='hello'
int(s)
錯誤舉例
AttributeError 試圖訪問一個物件沒有的樹形,比如foo.x,但是foo沒有屬性x
IOError 輸入/輸出異常;基本上是無法開啟檔案
ImportError 無法引入模組或包;基本上是路徑問題或名稱錯誤
IndentationError 語法錯誤(的子類) ;程式碼沒有正確對齊
IndexError 下標索引超出序列邊界,比如當x只有三個元素,卻試圖訪問x[5]
KeyError 試圖訪問字典裡不存在的鍵
KeyboardInterrupt Ctrl+C被按下
NameError 使用一個還未被賦予物件的變數
SyntaxError Python程式碼非法,程式碼不能編譯(個人認為這是語法錯誤,寫錯了)
TypeError 傳入物件型別與要求的不符合
UnboundLocalError 試圖訪問一個還未被設定的區域性變數,基本上是由於另有一個同名的全域性變數,
導致你以為正在訪問它
ValueError 傳入一個呼叫者不期望的值,即使值的型別是正確的
常用異常
ArithmeticError
AssertionError
AttributeError
BaseException
BufferError
BytesWarning
DeprecationWarning
EnvironmentError
EOFError
Exception
FloatingPointError
FutureWarning
GeneratorExit
ImportError
ImportWarning
IndentationError
IndexError
IOError
KeyboardInterrupt
KeyError
LookupError
MemoryError
NameError
NotImplementedError
OSError
OverflowError
PendingDeprecationWarning
ReferenceError
RuntimeError
RuntimeWarning
StandardError
StopIteration
SyntaxError
SyntaxWarning
SystemError
SystemExit
TabError
TypeError
UnboundLocalError
UnicodeDecodeError
UnicodeEncodeError
UnicodeError
UnicodeTranslateError
UnicodeWarning
UserWarning
ValueError
Warning
ZeroDivisionError
更多異常

二,異常處理

什麼是異常?

異常發生之後

異常之後的程式碼就不執行了

什麼是異常處理

python直譯器檢測到錯誤,觸發異常(也允許程式設計師自己觸發異常)

程式設計師編寫特定的程式碼,專門用來捕捉這個異常(這段程式碼與程式邏輯無關,與異常處理有關)

如果捕捉成功則進入另外一個處理分支,執行你為其定製的邏輯,使程式不會崩潰,這就是異常處理

為什麼要進行異常處理?

python解析器去執行程式,檢測到了一個錯誤時,觸發異常,異常觸發後且沒被處理的情況下,程式就在當前異常處終止,後面的程式碼不會執行,誰會去用一個執行著突然就崩潰的軟體。

所以你必須提供一種異常處理機制來增強你程式的健壯性與容錯性 

如何進行異常處理?

首先須知,異常是由程式的錯誤引起的,語法上的錯誤跟異常處理無關,必須在程式執行前就修正

一: 使用if判斷式

num1=input('>>: ') #輸入一個字串試試
int(num1)
正常程式碼
num1=input('>>: ') #輸入一個字串試試
if num1.isdigit():
    int(num1) #我們的正統程式放到了這裡,其餘的都屬於異常處理範疇
elif num1.isspace():
    print('輸入的是空格,就執行我這裡的邏輯')
elif len(num1) == 0:
    print('輸入的是空,就執行我這裡的邏輯')
else:
    print('其他情情況,執行我這裡的邏輯')

'''
問題一:
使用if的方式我們只為第一段程式碼加上了異常處理,但這些if,跟你的程式碼邏輯並無關係,這樣你的程式碼會因為可讀性差而不容易被看懂

問題二:
這只是我們程式碼中的一個小邏輯,如果類似的邏輯多,那麼每一次都需要判斷這些內容,就會倒置我們的程式碼特別冗長。
'''
使用if判斷進行異常處理

總結:

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

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]()
你之前用的異常處理機制

二:python異常處理的'私人定製'

python:為每一種異常定製了一個型別,然後提供了一種特定的語法結構用來進行異常處理

part1:基本語法

try:
     被檢測的程式碼塊
except 異常型別:
     try中一旦檢測到異常,就執行這個位置的邏輯
f = open('a.txt')

g = (line.strip() for line in f)
for line in g:
    print(line)
else:
    f.close()
讀檔案例1
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()
'''
讀檔案例2

part2:異常類只能用來處理指定的異常情況.

複製程式碼 複製程式碼
# 未捕獲到異常,程式直接報錯
 
s1 = 'hello'
try:
    int(s1)
except IndexError as e:
    print e
複製程式碼 複製程式碼

part3:多分支

複製程式碼 複製程式碼
s1 = 'hello'
try:
    int(s1)
except IndexError as e:
    print(e)
except KeyError as e:
    print(e)
except ValueError as e:
    print(e)
複製程式碼 複製程式碼

part4:萬能異常:Exception.

他可以捕獲任意異常,即:

s1 = 'hello'
try:
    int(s1)
except Exception as e:
    print(e)

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

你說的沒錯,但是應該分兩種情況去看

1.如果你想要的效果是,無論出現什麼異常,我們統一丟棄,或者使用同一段程式碼邏輯去處理他們,那麼騷年,大膽的去做吧,只有一個Exception就足夠了。

s1 = 'hello'
try:
    int(s1)
except Exception,e:
    '丟棄或者執行其他邏輯'
    print(e)

#如果你統一用Exception,沒錯,是可以捕捉所有異常,但意味著你在處理所有異常時都使用同一個邏輯去處理(這裡說的邏輯即當前expect下面跟的程式碼塊)
Exception

2.如果你想要的效果是,對於不同的異常我們需要定製不同的處理邏輯,那就需要用到多分支了。

s1 = 'hello'
try:
    int(s1)
except IndexError as e:
    print(e)
except KeyError as e:
    print(e)
except ValueError 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)
except Exception as e:
    print(e)
多分支+Exception

part5:異常的其他機構

複製程式碼 複製程式碼
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('無論異常與否,都會執行該模組,通常是進行清理工作')
複製程式碼 複製程式碼

part6:主動觸發異常

try:
    raise TypeError('型別錯誤')
except Exception as e:
    print(e)

part7:自定義異常

複製程式碼 複製程式碼
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)
複製程式碼 複製程式碼

part8:斷言

# assert 條件
 
assert 1 == 1
 
assert 1 == 2

part9:try..except的方式比較if的方式的好處

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

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

使用try..except的方式

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

什麼時候用異常處理

有的同學會這麼想,學完了異常處理後,好強大,我要為我的每一段程式都加上try...except,幹毛線去思考它會不會有邏輯錯誤啊,這樣就很好啊,多省腦細胞===》2B青年歡樂多

try...except應該儘量少用,因為它本身就是你附加給你的程式的一種異常處理的邏輯,與你的主要的工作是沒有關係的
這種東西加的多了,會導致你的程式碼可讀性變差,只有在有些異常無法預知的情況下,才應該加上try...except,其他的邏輯錯誤應該儘量修正