类有一个可定义的函数__exit__,可以实现上下文管理器.

它采用所需的参数:

def __exit__(self, exc_型, exc_val, exc_tb):

但我找不到关于这些论点是什么及其类型的明确定义.

以下是我对它们是什么以及为什么的最佳猜测,但我不完全确定:

def __exit__(self, exc_型: Exception, exc_val: TracebackException, exc_tb: TracebackType):

exc_型

Python定义了一个TracebackException类,该类接受exc_型参数,该参数在issubclassSyntaxError中的构造函数中上下文使用,从而推断exc_型实际上是Exception的某种类型,SyntaxError继承自Exception.

exc_val

另外,在TracebackException类中有一个exc_value参数,它与exc_val匹配,exc_val似乎有各种属性,比如__cause____context__,以及TracebackType本身定义的其他属性.这让我觉得这个参数本身就是TracebackException的一个实例.

exc_tb

Python定义了一个walk_tb函数,它使用exc_tb作为参数(从docs.python.org手动跟踪),这个对象似乎有tb_frametb_linenotb_next个属性,这些属性可以追溯到typeshed库中的TracebackType类.

思想?

推荐答案

exc_type是异常的类.exc_val是例外情况.exc_tb是一个回溯对象,types.TracebackType中有一个引用.

一般来说,情况应该是

  • type(exc_val) is exc_type
  • exc_val.__traceback__ is exc_tb

请注意,当上下文管理器下的代码没有引发异常时,仍然会调用__exit__,并且args将为(None, None, None),因此所有三个参数都应注释为optional.

那么它的正确注释应该如下所示:

def __exit__(self, exctype: Optional[Type[BaseException]],
             excinst: Optional[BaseException],
             exctb: Optional[TracebackType]) -> bool: ...

您可能想知道为什么这个API有三个参数,而其中两个参数可以从异常实例本身轻松确定.但情况并非总是这样,在较旧版本的Python中,可以将字符串作为异常引发,而异常的__traceback__属性并不存在.在Python2.7中,您仍然可以将旧式类作为例外提出(!)

Python-3.x相关问答推荐

这是重命名极地df列的最好方式吗?

如何将项目添加到Python中具有固定大小的列表列表中

Heroku 中的未知错误代码缺少一个或多个参数

requests.exceptions.InvalidSchema:未找到连接适配器.我试图遍历一个列表

Pandas 转换为日期时间

平移数组

将 rgb numpy 图像转换为 rgb 列表和相应的索引值

Pandas 在每组两个条件之间获得时间增量

如何将具有多个参数的函数传递给 python concurrent.futures.ProcessPoolExecutor.map()?

为什么 setattr 在绑定方法上失败

PyQt:退出时没有错误消息(回溯)

Python 3 list(dictionary.keys()) 引发错误.我究竟做错了什么?

str.format_map(mapping) 和 str.format 有什么区别

在计算之前删除包含某些值的组合

如何使用 asyncio 添加连接超时?

Asyncio RuntimeError:事件循环已关闭

如何将 Matplotlib 图形转换为 PIL Image 对象(不保存图像)

为什么排序列表比未排序列表大

如何避免使用我的 python 包构建 C 库?

新项目:Python 2 还是 Python 3?