• 8.5. with 语句

    8.5. with 语句

    with 语句用于包装带有使用上下文管理器 (参见 with 语句上下文管理器 一节) 定义的方法的代码块的执行。 这允许对普通的 tryexceptfinally 使用模式进行封装以方便地重用。

    1. with_stmt ::= "with" with_item ("," with_item)* ":" suite
    2. with_item ::= expression ["as" target]

    带有一个“项目”的 with 语句的执行过程如下:

    • 对上下文表达式 (在 with_item 中给出的表达式) 求值以获得一个上下文管理器。

    • 载入上下文管理器的 exit() 以便后续使用。

    • 发起调用上下文管理器的 enter() 方法。

    • 如果 with 语句中包含一个目标,来自 enter() 的返回值将被赋值给它。

    注解

    with 语句会保证如果 enter() 方法返回时未发生错误,则 exit() 将总是被调用。 因此,如果在对目标列表赋值期间发生错误,则会将其视为在语句体内部发生的错误。 参见下面的第 6 步。

    • 执行语句体。

    • 发起调用上下文管理器的 exit() 方法。 如果语句体的退出是由异常导致的,则其类型、值和回溯信息将被作为参数传递给 exit()。 否则的话,将提供三个 None 参数。

    如果语句体的退出是由异常导致的,并且来自 exit() 方法的返回值为假,则该异常会被重新引发。 如果返回值为真,则该异常会被抑制,并会继续执行 with 语句之后的语句。

    如果语句体由于异常以外的任何原因退出,则来自 exit() 的返回值会被忽略,并会在该类退出正常的发生位置继续执行。

    如果有多个项目,则会视作存在多个 with 语句嵌套来处理多个上下文管理器:

    1. with A() as a, B() as b:
    2. suite

    等价于

    1. with A() as a:
    2. with B() as b:
    3. suite

    在 3.1 版更改: 支持多个上下文表达式。

    参见

    • PEP 343 - "with" 语句
    • Python with 语句的规范描述、背景和示例。