# 处理异常
# 处理异常
我们还没有谈到`__exit__`方法的这三个参数:`type`, `value`和`traceback`。
在第4步和第6步之间,如果发生异常,Python会将异常的`type`,`value`和`traceback`传递到`__exit__`方法。
它让`__exit__`方法来决定如何关闭文件以及是否需要其他步骤。在我们的案例中,我们并没有注意它们。
那如果我们的文件对象抛出一个异常呢?万一我们尝试访问文件对象的一个不支持的方法。举个例子:
~~~
with File('demo.txt', 'w') as opened_file:
opened_file.undefined_function('Hola!')
~~~
我们来列一下,当异常发生时,`with`语句会采取哪些步骤。
1. 它把异常的`type`,`value`和`traceback`传递给`__exit__`方法
1. 它让`__exit__`方法来处理异常
1. 如果`__exit__`返回的是True,那么这个异常就被优雅地处理了。
1. 如果`__exit__`返回的是True以外的任何东西,那么这个异常将被`with`语句抛出。
在我们的案例中,`__exit__`方法返回的是`None`(如果没有`return`语句那么方法会返回`None`)。因此,`with`语句抛出了那个异常。
~~~
Traceback (most recent call last):
File "<stdin>", line 2, in <module>
AttributeError: 'file' object has no attribute 'undefined_function'
~~~
我们尝试下在`__exit__`方法中处理异常:
~~~
class File(object):
def __init__(self, file_name, method):
self.file_obj = open(file_name, method)
def __enter__(self):
return self.file_obj
def __exit__(self, type, value, traceback):
print("Exception has been handled")
self.file_obj.close()
return True
with File('demo.txt', 'w') as opened_file:
opened_file.undefined_function()
# Output: Exception has been handled
~~~
我们的`__exit__`方法返回了`True`,因此没有异常会被`with`语句抛出。
这还不是实现上下文管理器的唯一方式。还有一种方式,我们会在下一节中一起看看。
- 简介
- 序
- 译后感
- 原作者前言
- *args 和 **kwargs
- *args 的用法
- **kwargs 的用法
- 使用 *args 和 **kwargs 来调用函数
- 啥时候使用它们
- 调试 Debugging
- 生成器 Generators
- 可迭代对象(Iterable)
- 迭代器(Iterator)
- 迭代(Iteration)
- 生成器(Generators)
- Map和Filter
- Map
- Filter
- set 数据结构
- 三元运算符
- 装饰器
- 一切皆对象
- 在函数中定义函数
- 从函数中返回函数
- 将函数作为参数传给另一个函数
- 你的第一个装饰器
- 使用场景
- 授权
- 日志
- 带参数的装饰器
- 在函数中嵌入装饰器
- 装饰器类
- Global和Return
- 多个return值
- 对象变动 Mutation
- slots魔法
- 虚拟环境
- 容器 Collections
- 枚举 Enumerate
- 对象自省
- dir
- type和id
- inspect模块
- 推导式 Comprehension
- 列表推导式
- 字典推导式
- 集合推导式
- 异常
- 处理多个异常
- finally从句
- try/else从句
- lambda表达式
- 一行式
- For - Else
- else语句
- open函数
- 目标Python2+3
- 协程
- 函数缓存
- Python 3.2+
- Python 2+
- 上下文管理器
- 基于类的实现
- 处理异常
- 基于生成器的实现