# 處理異常
# 處理異常
我們還沒有談到`__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+
- 上下文管理器
- 基于類的實現
- 處理異常
- 基于生成器的實現