异常处理
异常处理
一 异常介绍
1.1什么是异常
异常是程序发生错误的信号。程序一旦出现错误,便会产生一个异常,若程序中没有处理它,就会抛出该异常,程序的运行也随之终止。在Python中,错误触发的异常如下
1.2 为何要处理异常
为了增强程序的健壮性,即便是程序运行过程中出错了,也不要终止程序,而是捕捉异常并处理:将出错信息记录到日志内
二 如何处理异常
2.1 错误的两种方式
而错误分成两种,一种是语法上的错误SyntaxError,这种错误应该在程序运行前就修改正确
if 1>3
print("run...")
另一类就是逻辑错误,常见的逻辑错误如
# TypeError:数字类型无法与字符串类型相加
1+’2’
# ValueError:当字符串包含有非数字的值时,无法转成int类型
num=input(">>: ") #输入hello
int(num)
# NameError:引用了一个不存在的名字x
x
# IndexError:索引超出列表的限制
l=['egon','aa']
l[3]
# KeyError:引用了一个不存在的key
dic={'name':'egon'}
dic['age']
# AttributeError:引用的属性不存在
class Foo:
pass
Foo.x
# ZeroDivisionError:除数不能为0
1/0
2.2 逻辑错误的两种处理方式
2.2.1误发生的条件是可以预知的
#使用if判断来解决
age=input('>>: ').strip() # 输入的只要不是数字就会出错
if age.isdigit():
age=int(age)
if age > 18:
print('猜大了')
elif age < 18:
print('猜大了')
else:
print('猜对了')
else:
print('必须输入数字')
2.2.2 错误条件是无法预知的
try:
# 有可能会抛出异常的代码
子代码1
子代码2
子代码3
except 异常类型1 as e:
pass
except 异常类型2 as e:
pass
...
else:
如果被检测的子代码块没有异常发生,则会执行else的子代码,else不能单独与try使用,必须要搭配except
finally:
无论被检测的子代码块有无异常发生,都会执行finally的子代码,finally可以单独和try使用,finally结束后依旧抛出异常,之后的代码都不会执行,因此finally的子代码主要做回收资源的操作
用法一:
try:
print('1111111111')
l=['aaa','bbbb']
l[3] # 抛出异常IndexError,该行代码同级别的后续代码不会运行
print('2222222222')
xxx
print('33333333')
dic={'a':1}
dic['a']
except IndexError as e:
print('异常的信息: ',e)
except NameError as e:
print('异常的信息: ',e)
用法二:异常处理方式相同时
try:
print('1111111111')
l = ['aaa', 'bbbb']
l[3] # 抛出异常IndexError,该行代码同级别的后续代码不会运行
print('2222222222')
xxx
print('33333333')
dic = {'a': 1}
dic['aaa']
except (IndexError, NameError) as e:
print('异常的信息: ', e)
except KeyError as e:
print('字典的key不存在: ', e)
用法三:万能处理异常
try:
print('1111111111')
l = ['aaa', 'bbbb']
#l[3] # 抛出异常IndexError,该行代码同级别的后续代码不会运行
print('2222222222')
xxx
print('33333333')
dic = {'a': 1}
dic['aaa']
except Exception as e: # 万能异常
print('所有异常都可以匹配的到')
在前面章节的学习中,遗留过一个问题,即是否可以在程序的指定位置手动抛出一个异常?答案是肯定的,Python 允许我们在程序中手动设置异常,使用 raise 语句即可。 读者可能会感到疑惑,即我们从来都是想方设法地让程序正常运行,为什么还要手动设置异常呢?首先要分清楚程序发生异常和程序执行错误,它们完全是两码事,程序由于错误导致的运行异常,是需要程序员想办法解决的;但还有一些异常,是程序正常运行的结果,比如用 raise 手动引发的异常。 raise 语句的基本语法格式为: raise [exceptionName [(reason)]] 其中,用 [] 括起来的为可选参数,其作用是指定抛出的异常名称,以及异常信息的相关描述。如果可选参数全部省略,则 raise 会把当前错误原样抛出;如果仅省略 (reason),则在抛出异常时,将不附带任何的异常描述信息。 也就是说,raise 语句有如下三种常用的用法: raise:单独一个 raise。该语句引发当前上下文中捕获的异常(比如在 except 块中),或默认引发 RuntimeError 异常。 raise 异常类名称:raise 后带一个异常类名称,表示引发执行类型的异常。 raise 异常类名称(描述信息):在引发指定类型的异常的同时,附带异常的描述信息。 想了解一下常用的异常类名称,可以阅读《Python常见异常类型》一节。 显然,每次执行 raise 语句,都只能引发一次执行的异常。首先,我们来测试一下以上 3 种 raise 的用法: >>> raise Traceback (most recent call last): File "<pyshell#1>", line 1, in <module> raise RuntimeError: No active exception to reraise >>> raise ZeroDivisionError Traceback (most recent call last): File "<pyshell#0>", line 1, in <module> raise ZeroDivisionError ZeroDivisionError >>> raise ZeroDivisionError("除数不能为零") Traceback (most recent call last): File "<pyshell#2>", line 1, in <module> raise ZeroDivisionError("除数不能为零") ZeroDivisionError: 除数不能为零 当然,我们手动让程序引发异常,很多时候并不是为了让其崩溃。事实上,raise 语句引发的异常通常用 try except(else finally)异常处理结构来捕获并进行处理。例如: try: a = input("输入一个数:") #判断用户输入的是否为数字 if(not a.isdigit()): raise ValueError("a 必须是数字") except ValueError as e: print("引发异常:",repr(e)) 程序运行结果为: 输入一个数:a 引发异常: ValueError('a 必须是数字',) 可以看到,当用户输入的不是数字时,程序会进入 if 判断语句,并执行 raise 引发 ValueError 异常。但由于其位于 try 块中,因为 raise 抛出的异常会被 try 捕获,并由 except 块进行处理。 因此,虽然程序中使用了 raise 语句引发异常,但程序的执行是正常的,手动抛出的异常并不会导致程序崩溃。 raise 不需要参数 正如前面所看到的,在使用 raise 语句时可以不带参数,例如: try: a = input("输入一个数:") if(not a.isdigit()): raise ValueError("a 必须是数字") except ValueError as e: print("引发异常:",repr(e)) raise 程序执行结果为: 输入一个数:a 引发异常: ValueError('a 必须是数字',) Traceback (most recent call last): File "D:\python3.6\1.py", line 4, in <module> raise ValueError("a 必须是数字") ValueError: a 必须是数字 这里重点关注位于 except 块中的 raise,由于在其之前我们已经手动引发了 ValueError 异常,因此这里当再使用 raise 语句时,它会再次引发一次。 当在没有引发过异常的程序使用无参的 raise 语句时,它默认引发的是 RuntimeError 异常。例如: try: a = input("输入一个数:") if(not a.isdigit()): raise except RuntimeError as e: print("引发异常:",repr(e)) 程序执行结果为: 输入一个数:a 引发异常: RuntimeError('No active exception to reraise',)
遗忘的知识点(主动抛出异常 raise 异常类型(‘’报错内容‘’))



浙公网安备 33010602011771号