Python - 异常链


异常链是一种处理异常的技术,方法是在将捕获的异常包装在新异常中后重新抛出该异常。原始异常被保存为新异常的属性(例如原因)。

在处理一个异常“A”期间,可能会发生另一异常“B”。了解这两个异常对于调试问题很有用。有时,异常处理程序故意重新引发异常很有用,可以提供额外的信息,也可以将异常转换为另一种类型。

在Python 3.x中,可以实现异常链。如果 except 部分中有任何未处理的异常,则会将正在处理的异常附加到它并包含在错误消息中。

例子

在以下代码片段中,尝试打开不存在的文件会引发 FileNotFoundError。它由 except 块检测到。在处理另一个异常时引发。

try:
   open("nofile.txt")
except OSError:
   raise RuntimeError("unable to handle error")

它将产生以下输出-

Traceback (most recent call last):
  File "/home/cg/root/64afcad39c651/main.py", line 2, in <module>
open("nofile.txt")
FileNotFoundError: [Errno 2] No such file or directory: 'nofile.txt'

During handling of the above exception, another exception occurred:

Traceback (most recent call last):
  File "/home/cg/root/64afcad39c651/main.py", line 4, in <module>
    raise RuntimeError("unable to handle error")
RuntimeError: unable to handle error

增加 。。从

如果在 raise 语句中使用可选的 from 子句,则表明一个异常是另一个异常的直接结果。当您转换异常时,这可能很有用。from 关键字后面的标记应该是异常对象。

try:
   open("nofile.txt")
except OSError as exc:
   raise RuntimeError from exc

它将产生以下输出-

Traceback (most recent call last):
  File "/home/cg/root/64afcad39c651/main.py", line 2, in <module>
    open("nofile.txt")
FileNotFoundError: [Errno 2] No such file or directory: 'nofile.txt'

The above exception was the direct cause of the following exception:

Traceback (most recent call last):
  File "/home/cg/root/64afcad39c651/main.py", line 4, in <module>
    raise RuntimeError from exc
RuntimeError

增加 。。从无

如果我们在 from 子句中使用 None 而不是异常对象,则前面示例中发现的自动异常链接将被禁用。

try:
   open("nofile.txt")
except OSError as exc:
   raise RuntimeError from None

它将产生以下输出-

Traceback (most recent call last):
 File "C:\Python311\hello.py", line 4, in <module>
  raise RuntimeError from None
RuntimeError

__背景__和__原因__

在 except 块中引发异常会自动将捕获的异常添加到新异常的 __context__ 属性中。同样,您还可以使用表达式raise ... from语法将 __cause__ 添加到任何异常。

try:
   try:
      raise ValueError("ValueError")
   except ValueError as e1:
      raise TypeError("TypeError") from e1
except TypeError as e2:
   print("The exception was", repr(e2))
   print("Its __context__ was", repr(e2.__context__))
   print("Its __cause__ was", repr(e2.__cause__))

它将产生以下输出-

The exception was TypeError('TypeError')
Its __context__ was ValueError('ValueError')
Its __cause__ was ValueError('ValueError')