为什么在“except”块后使用命名异常会得到“NameError”(或“UnboundLocalError”)?
- 2025-01-14 08:50:00
- admin 原创
- 102
问题描述:
此示例代码在 2.x 中有效:
exc = None
try:
raise Exception
except Exception as exc:
pass
print(exc)
但在 3.x 中我收到一条错误消息NameError: name 'exc' is not defined
。如果我将代码放入函数中,则会收到错误消息UnboundLocalError: local variable 'exc' referenced before assignment
。
我想也许我可以通过明确的任务来解决这个问题,就像这样:
exc = None
try:
raise Exception
except Exception as exc:
exc = exc
但这不起作用。为什么?我如何从块外部访问异常except
?
解决方案 1:
该try
语句明确限制了绑定异常的范围,以防止循环引用导致其泄漏。
当使用目标指定异常时,它将在 except 子句的末尾被清除。
[...]
这意味着必须将异常分配给不同的名称,以便能够在 except 子句之后引用它。异常被清除是因为在附加了回溯的情况下,它们与堆栈框架形成了一个引用循环,使该框架中的所有局部变量保持活动状态,直到下一次垃圾回收发生。
强调我的;为了随后使用异常,必须将其绑定到新名称。exc = exc
不会有帮助,因为没有创建新的范围,而是从范围中删除语句except
中指定的名称。except
在 Python 2 中,异常没有对回溯的引用,因此这种清除对于防止垃圾收集问题来说不是必要的。现在有了,所以它改变了。
然而,即使在 Python 2 中,你也会被明确警告清理回溯:
警告:将回溯返回值赋给处理异常的函数中的局部变量将导致循环引用。这将阻止同一函数中的局部变量或回溯引用的任何内容被垃圾收集。由于大多数函数不需要访问回溯,因此最好的解决方案是使用类似的东西
exctype, value = sys.exc_info()[:2]
来仅提取异常类型和值。如果您确实需要回溯,请确保在使用后将其删除(最好使用try ... finally
语句完成)或调用exc_info()
本身不处理异常的函数。
如果您重新绑定异常,您可能需要明确清除回溯:
try:
raise Exception("foo")
except Exception as e:
exc = e
exc.__traceback__ = None