# 協程
# 23. 協程
Python中的協程和生成器很相似但又稍有不同。主要區別在于:
- 生成器是數據的生產者
- 協程則是數據的消費者
首先我們先來回顧下生成器的創建過程。我們可以這樣去創建一個生成器:
~~~
def fib():
a, b = 0, 1
while True:
yield a
a, b = b, a+b
~~~
然后我們經常在`for`循環中這樣使用它:
~~~
for i in fib():
print i
~~~
這樣做不僅快而且不會給內存帶來壓力,因為我們所需要的值都是動態生成的而不是將他們存儲在一個列表中。更概括的說如果現在我們在上面的例子中使用`yield`便可獲得了一個協程。協程會消費掉發送給它的值。Python實現的`grep`就是個很好的例子:
~~~
def grep(pattern):
print("Searching for", parttern)
while True:
line = (yield)
if pattern in line:
print(line)
~~~
等等!`yield`返回了什么?啊哈,我們已經把它變成了一個協程。它將不再包含任何初始值,相反要從外部傳值給它。我們可以通過`send()`方法向它傳值。這有個例子:
~~~
search = grep('coroutine')
next(search)
#output: Searching for coroutine
search.send("I love you")
search.send("Don't you love me?")
search.send("I love coroutine instead!")
#output: I love coroutine instead!
~~~
發送的值會被`yield`接收。我們為什么要運行`next()`方法呢?這樣做正是為了啟動一個協程。就像協程中包含的生成器并不是立刻執行,而是通過`next()`方法來響應`send()`方法。因此,你必須通過`next()`方法來執行`yield`表達式。
我們可以通過調用`close()`方法來關閉一個協程。像這樣:
~~~
search = grep('coroutine')
search.close()
~~~
更多協程相關知識的學習大家可以參考David Beazley的這份[精彩演講](http://www.dabeaz.com/coroutines/Coroutines.pdf)。
- 簡介
- 序
- 譯后感
- 原作者前言
- *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+
- 上下文管理器
- 基于類的實現
- 處理異常
- 基于生成器的實現