语句的工作原理如下:

    • 首先,执行 try 子句try 和 关键字之间的(多行)语句)。

    • 如果没有异常发生,则跳过 except 子句 并完成 try 语句的执行。

    • 如果在执行 try 子句时发生了异常,则跳过该子句中剩下的部分。 然后,如果异常的类型和 关键字后面的异常匹配,则执行 except 子句,然后继续执行 try 语句之后的代码。

    一个 语句可能有多个 except 子句,以指定不同异常的处理程序。 最多会执行一个处理程序。 处理程序只处理相应的 try 子句中发生的异常,而不处理同一 try 语句内其他处理程序中的异常。 一个 except 子句可以将多个异常命名为带括号的元组,例如:

    1. ... except (RuntimeError, TypeError, NameError):
    2. ... pass

    如果发生的异常和 except 子句中的类是同一个类或者是它的基类,则异常和 except 子句中的类是兼容的(但反过来则不成立 —- 列出派生类的 except 子句与基类不兼容)。 例如,下面的代码将依次打印 B, C, D

    请注意如果 except 子句被颠倒(把 except B 放到第一个),它将打印 B,B,B —- 即第一个匹配的 except 子句被触发。

    最后的 except 子句可以省略异常名,以用作通配符。但请谨慎使用,因为以这种方式很容易掩盖真正的编程错误!它还可用于打印错误消息,然后重新引发异常(同样允许调用者处理异常):

    1. import sys
    2. try:
    3. s = f.readline()
    4. except OSError as err:
    5. print("OS error: {0}".format(err))
    6. except ValueError:
    7. print("Could not convert data to an integer.")
    8. except:
    9. print("Unexpected error:", sys.exc_info()[0])
    10. raise

    使用 else 子句比向 子句添加额外的代码要好,因为它避免了意外捕获由 tryexcept 语句保护的代码未引发的异常。

    发生异常时,它可能具有关联值,也称为异常 参数 。参数的存在和类型取决于异常类型。

    except 子句可以在异常名称后面指定一个变量。这个变量和一个异常实例绑定,它的参数存储在 instance.args 中。为了方便起见,异常实例定义了 ,因此可以直接打印参数而无需引用 .args 。也可以在抛出之前首先实例化异常,并根据需要向其添加任何属性。:

    1. >>> try:
    2. ... raise Exception('spam', 'eggs')
    3. ... except Exception as inst:
    4. ... print(type(inst)) # the exception instance
    5. ... print(inst) # __str__ allows args to be printed directly,
    6. ... # but may be overridden in exception subclasses
    7. ... x, y = inst.args # unpack args
    8. ... print('x =', x)
    9. ... print('y =', y)
    10. ...
    11. <class 'Exception'>
    12. ('spam', 'eggs')
    13. ('spam', 'eggs')
    14. x = spam

    如果异常有参数,则它们将作为未处理异常的消息的最后一部分(’详细信息’)打印。