# 基于生成器的實現
# 基于生成器的實現
我們還可以基于裝飾器(decorators)和生成器(generators)來實現上下文管理器。
Python有個`contextlib`模塊專門用于這個目的。我們可以使用一個生成器函數來實現一個上下文管理器,而不是使用一個類。
讓我們看看一個基本的,沒用的例子:
~~~
from contextlib import contextmanager
@contextmanager
def open_file(name):
f = open(name, 'w')
yield f
f.close()
~~~
OK啦!這個實現方式看起來更加直觀和簡單。然而,這個方法需要關于生成器、`yield`和裝飾器的一些額外知識。在這個例子中我們還沒有捕捉可能產生的任何異常。它的工作方式和之前的方法是大部分相同的。
讓我們小小地剖析下這個方法。
1. Python解釋器遇到了`yield`關鍵字。因為這個緣故它創建了一個生成器而不是一個普通的函數。
1. 因為這個裝飾器,`contextmanager`會被調用并傳入函數名(`open_file`)作為參數。
1. `contextmanager`函數返回一個以`GeneratorContextManager`對象封裝過的生成器。
1. 這個`GeneratorContextManager`被賦值給`open_file`函數,我們實際上是在調用`GeneratorContextManager`對象。
那現在我們既然知道了所有這些,我們可以用這個新生成的上下文管理器了,像這樣:
~~~
with open_file('some_file') as f:
f.write('hola!')
~~~
- 簡介
- 序
- 譯后感
- 原作者前言
- *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+
- 上下文管理器
- 基于類的實現
- 處理異常
- 基于生成器的實現