语句的工作原理如下:
首先,执行 try 子句 (
try
和 关键字之间的(多行)语句)。如果没有异常发生,则跳过 except 子句 并完成
try
语句的执行。如果在执行 try 子句时发生了异常,则跳过该子句中剩下的部分。 然后,如果异常的类型和 关键字后面的异常匹配,则执行 except 子句,然后继续执行
try
语句之后的代码。
一个 语句可能有多个 except 子句,以指定不同异常的处理程序。 最多会执行一个处理程序。 处理程序只处理相应的 try 子句中发生的异常,而不处理同一 try
语句内其他处理程序中的异常。 一个 except 子句可以将多个异常命名为带括号的元组,例如:
... except (RuntimeError, TypeError, NameError):
... pass
如果发生的异常和 except
子句中的类是同一个类或者是它的基类,则异常和 except 子句中的类是兼容的(但反过来则不成立 —- 列出派生类的 except 子句与基类不兼容)。 例如,下面的代码将依次打印 B, C, D
请注意如果 except 子句被颠倒(把 except B
放到第一个),它将打印 B,B,B —- 即第一个匹配的 except 子句被触发。
最后的 except 子句可以省略异常名,以用作通配符。但请谨慎使用,因为以这种方式很容易掩盖真正的编程错误!它还可用于打印错误消息,然后重新引发异常(同样允许调用者处理异常):
import sys
try:
s = f.readline()
except OSError as err:
print("OS error: {0}".format(err))
except ValueError:
print("Could not convert data to an integer.")
except:
print("Unexpected error:", sys.exc_info()[0])
raise
使用 else
子句比向 子句添加额外的代码要好,因为它避免了意外捕获由 try
… except
语句保护的代码未引发的异常。
发生异常时,它可能具有关联值,也称为异常 参数 。参数的存在和类型取决于异常类型。
except 子句可以在异常名称后面指定一个变量。这个变量和一个异常实例绑定,它的参数存储在 instance.args
中。为了方便起见,异常实例定义了 ,因此可以直接打印参数而无需引用 .args
。也可以在抛出之前首先实例化异常,并根据需要向其添加任何属性。:
>>> try:
... raise Exception('spam', 'eggs')
... except Exception as inst:
... print(type(inst)) # the exception instance
... print(inst) # __str__ allows args to be printed directly,
... # but may be overridden in exception subclasses
... x, y = inst.args # unpack args
... print('x =', x)
... print('y =', y)
...
<class 'Exception'>
('spam', 'eggs')
('spam', 'eggs')
x = spam
如果异常有参数,则它们将作为未处理异常的消息的最后一部分(’详细信息’)打印。