备注:本人在转载这篇文章以后,还做了一些修改,增加了自己的一些代码和理解,以黄色背景色标示出来。

Python raise用法

是否可以在程序的指定位置手动抛出一个异常?答案是肯定的,Python 允许我们在程序中手动设置异常,使用 raise 语句即可。

读者可能会感到疑惑,即我们从来都是想方设法地让程序正常运行,为什么还要手动设置异常呢?首先要分清楚程序发生异常和程序执行错误,它们完全是两码事,程序由于错误导致的运行异常,是需要程序员想办法解决的;但还有一些异常,是程序正常运行的结果,比如用 raise 手动引发的异常。

raise 语句的基本语法格式为:

raise [exceptionName [(reason)]]

其中,用 [] 括起来的为可选参数,其作用是指定抛出的异常名称,以及异常信息的相关描述。如果可选参数全部省略,则 raise 会把当前错误原样抛出;如果仅省略 (reason),则在抛出异常时,将不附带任何的异常描述信息。

也就是说,raise 语句有如下三种常用的用法:

  1. raise:单独一个 raise。该语句引发当前上下文中捕获的异常(比如在 except 块中),或默认引发 RuntimeError 异常。
  2. raise 异常类名称:raise 后带一个异常类名称,表示引发执行类型的异常。
  3. raise 异常类名称(描述信息):在引发指定类型的异常的同时,附带异常的描述信息。

想了解一下常用的异常类名称,可以阅读《Python常见异常类型》一节。

链接为:什么是异常处理,Python常见异常类型(入门必读)开发人员在编写程序时,难免会遇到错误,有的是编写人员疏忽造成的语法错误,有的是程序内部隐含逻辑问题造成的数据错误,还有的是程序运行时与系统的规则冲突造成的系统错误http://c.biancheng.net/view/4593.html

显然,每次执行 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
"""