异常
什么是异常?
在数学里面,我们都知道,除数是不能为0的,在Python里面也是一样的,所以,以下语句是有问题的
result = 1 / 0
当我们运行这个语句,会得到一个报错信息
这个报错就是一个异常。除此之外,如果之前学习的过程中有实际敲代码,你应该遇到过各种异常,比如获取列表数据时的索引超出范围(IndexError: list index out of range
)、获取字典数据时没有指定的键(KeyError: 'xxx'
)等等。
一般情况我们会想办法在代码里面判断有没有可能出现异常,要尽量避免出现异常报错,但是有时候总要考虑不完善的时候,或者我们就是要根据异常来做下一步处理,这个时候我们就要在异常出现的地方进行捕捉了。
异常捕捉
异常捕捉在Python里面用try...except...
的结构处理
try...except...语句
try:
可能出现异常的执行语句
except Exception:
异常发生后的处理语句
例如:
try:
result = 1 / 0
except Exception:
print("发生了异常")
上面例子执行后会输出发生了异常
,因为 1/0
肯定会触发异常,但是这个异常被我们捕捉到了,然后执行了下面的输出语句。
如果想要知道具体发生了上面异常,可以将异常放入一个变量,然后在下面打印
try:
result = 1 / 0
except Exception as e:
print(f"发生了异常: {e}")
捕捉指定异常
上面例子中,我们是直接捕捉所以异常,但是这种眉毛胡子一把抓的方式往往不适用,我们更希望针对指定的异常类型进行处理,对于其他的异常,我们还是希望它能报出来,便于我们改进代码.
指定捕捉异常类型,我们在except后面添加异常类型即可,看个例子
try:
result = 1 / 0
except ZeroDivisionError: # 指定捕捉除数为0的异常
print("计算中的除数为0了")
也可以同时捕捉多个异常,只需在后面继续用except捕捉即可
numbers = [4, 5, 6]
something = {'div': 0}
try:
result = numbers[2] / something['div']
except ZeroDivisionError: # 指定捕捉除数为0的异常
print("发生了异常")
except KeyError:
print("字典取值错误")
except IndexError:
print("列表索引超出范围")
except Exception as e:
print(f"其他异常:{e}")
else和finally
在try...except...后面可以跟上else或finally或者都用上。
- try...except...else...:如果程序执行过程中没有异常,就会执行else后面的语句
try:
print("输出语句,肯定不会报错!")
except Exception:
print("不可能执行的")
else:
print("哇哦~没有报错耶!")
- try...except...finally:不管上面的程序有没有报错,最后都会执行finally语句
try:
result = 1 / 0
except Exception:
print("报错了!")
else:
print("哇哦~没有报错耶!")
finally:
print("最后都会执行的")
运行结果如下,即输出了异常捕捉的处理语句,又输出了finally的处理语句
报错了!
最后都会执行的
主动抛异常
有时候异常不一定是意料之外产生的,也可能是我们主动抛出来的。例如:有一个函数,接受两个参数,要求传入两个浮点类型的参数做运算,但是传入的参数是字符串类型的,这种时候,我们就可以直接抛出异常了,不需要让程序继续执行下去,等到程序报错。
主动报错可以使用关键词raise,后面加上错误类型,类型里面可以填写提示信息,例如:
# 这个函数必须接收两个浮点数
def do_something(a, b):
if not isinstance(a, float) or not isinstance(b, float):
raise ValueError("请传入两个浮点数!") # 在这里抛出错误
result = a * b * 3.1415926
return result
例如上面的例子,在参数传递到函数的时候就先判断函数的数据类型是不是浮点型,如果不是浮点型,那么就没有必要继续执行了,直接主动抛出一个异常,提示用户传参类型错误!
结语
本章我们学习了对于异常的处理,使用异常处理后,我们可以在可能出现异常的地方做异常捕捉,这样可以使程序更稳定运行,大大增强了程序的稳健性。当然,如果程序本身逻辑很简单,事先能判断出可能出现的异常并处理,确保程序一定不会报错,那么不使用异常捕捉也是可以的。