备注:本人在转载这篇文章以后,还做了一些修改,增加了自己的一些代码和理解,以黄色背景色标示出来。
Python raise用法
是否可以在程序的指定位置手动抛出一个异常?答案是肯定的,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代码块里发生的异常,会被抛出来,供except捕获,捕获到的异常在except代码块中处理。)例如:
try:
a = input("输入一个数:")
# 判断用户输入的是否为数字
if (not a.isdigit()):
raise ValueError("a 必须是数字")
except ValueError as e:
print("引发异常:", repr(e))
""" Output: 输入一个数:d 引发异常: ValueError('a 必须是数字') Process finished with exit code 0 """
可以看到,当用户输入的不是数字时,程序会进入 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
"""
Output:
输入一个数:d
引发异常: ValueError('a 必须是数字')
Traceback (most recent call last):
File "E:\workspace\PycharmProjects\project01\com\study\pythoncore\ch02\test03.py", line 17, in <module>
raise ValueError("a 必须是数字")
ValueError: a 必须是数字
Process finished with exit code 1
"""
这里重点关注位于 except 块中的 raise,由于在其之前我们已经手动引发了 ValueError 异常,因此这里当再使用 raise 语句时,它会再次引发一次。(本人的话:在except代码块中,由raise引发的异常,代码没有做处理,所以导致了程序崩溃,即:Process finished with exit code 1)
当在没有引发过异常的程序使用无参的 raise 语句时,它默认引发的是 RuntimeError 异常。(本人的话:由于except代码块没有针对RuntimeError的捕获,所以导致了程序崩溃。 )例如:
try:
a = input("输入一个数:")
# 判断用户输入的是否为数字
if (not a.isdigit()):
raise
except ValueError as e:
print("引发异常:", repr(e))
"""
Output:
输入一个数:d
Traceback (most recent call last):
File "E:\workspace\PycharmProjects\project01\com\study\pythoncore\ch02\test03.py", line 17, in <module>
raise
RuntimeError: No active exception to reraise
Process finished with exit code 1
"""
把程序修改后:
try:
a = input("输入一个数:")
# 判断用户输入的是否为数字
if (not a.isdigit()):
raise
except ValueError as e:
print("引发异常:", repr(e))
except RuntimeError as e:
print("引发异常:", repr(e))
"""
Output:
输入一个数:d
引发异常: RuntimeError('No active exception to reraise')
Process finished with exit code 0
"""
本人的话:当exception被捕获,程序就会正常退出(即:Process finished with exit code 0)。没被捕获,程序就会崩溃(即:Process finished with exit code 1)。
finally的作用是:无论代码是否抛出异常,程序在正常退出或者崩溃之前,都会执行finaly里的代码块。代码示例如下:
try:
a = input("输入一个数:")
# 判断用户输入的是否为数字
if (not a.isdigit()):
raise
except ValueError as e:
print("引发异常:", repr(e))
except RuntimeError as e:
print("引发异常:", repr(e))
finally:
print("程序执行到finally代码块里。")
"""
捕获到正确对口的异常以后,输出结果如下:
except RuntimeError as e:
print("引发异常:", repr(e))
Output:
输入一个数:d
引发异常: RuntimeError('No active exception to reraise')
程序执行到finally代码块里。
Process finished with exit code 0
----
注释掉捕获到正确对口的异常以后,输出结果如下:
# except RuntimeError as e:
# print("引发异常:", repr(e))
Output:
输入一个数:d
程序执行到finally代码块里。
Traceback (most recent call last):
File "E:\workspace\PycharmProjects\project01\com\study\pythoncore\ch02\test03.py", line 17, in <module>
raise
RuntimeError: No active exception to reraise
Process finished with exit code 1
"""