鍍金池/ 教程/ Python/ 23. 協(xié)程
<code>open</code>函數(shù)
Python 2系列版本
可迭代對(duì)象(Iterable)
異常
在函數(shù)中嵌入裝飾器
你的第一個(gè)裝飾器
上下文管理器(Context managers)
<code>set</code>(集合)數(shù)據(jù)結(jié)構(gòu)
裝飾器類
字典推導(dǎo)式(<code>dict</code> comprehensions)
<code>Reduce</code>
捐贈(zèng)名單
<code>Filter</code>
<code>try/else</code>從句
*args 的用法
<code>dir</code>
處理異常
<code>else</code>從句
對(duì)象自省
For - Else
18. 一行式
Python 3.2及以后版本
Global和Return
基于類的實(shí)現(xiàn)
容器(<code>Collections</code>)
23. 協(xié)程
推薦閱讀
譯者后記
<code>*args</code> 和 <code>**kwargs</code>
**kwargs 的用法
生成器(Generators)
迭代(Iteration)
基于生成器的實(shí)現(xiàn)
將函數(shù)作為參數(shù)傳給另一個(gè)函數(shù)
日志(Logging)
三元運(yùn)算符
<code>inspect</code>模塊
枚舉
Map,F(xiàn)ilter 和 Reduce
各種推導(dǎo)式(comprehensions)
從函數(shù)中返回函數(shù)
列表推導(dǎo)式(<code>list</code> comprehensions)
處理多個(gè)異常
帶參數(shù)的裝飾器
對(duì)象變動(dòng)(Mutation)
22. 目標(biāo)Python2+3
迭代器(Iterator)
虛擬環(huán)境(virtualenv)
<code>__slots__</code>魔法
什么時(shí)候使用它們?
Python/C API
<code>Map</code>
SWIG
授權(quán)(Authorization)
裝飾器
一切皆對(duì)象
使用C擴(kuò)展
使用 <code>*args</code> 和 <code>**kwargs</code> 來調(diào)用函數(shù)
17. <code>lambda</code>表達(dá)式
集合推導(dǎo)式(<code>set</code> comprehensions)
<code>type</code>和<code>id</code>
在函數(shù)中定義函數(shù)
<code>finally</code>從句
CTypes
調(diào)試(Debugging)
使用場(chǎng)景
生成器(Generators)
多個(gè)return值
關(guān)于原作者
函數(shù)緩存 (Function caching)
Python進(jìn)階

23. 協(xié)程

Python中的協(xié)程和生成器很相似但又稍有不同。主要區(qū)別在于:

  • 生成器是數(shù)據(jù)的生產(chǎn)者
  • 協(xié)程則是數(shù)據(jù)的消費(fèi)者

首先我們先來回顧下生成器的創(chuàng)建過程。我們可以這樣去創(chuàng)建一個(gè)生成器:

    def fib():
        a, b = 0, 1
        while True:
            yield a
            a, b = b, a+b

然后我們經(jīng)常在for循環(huán)中這樣使用它:

    for i in fib():
        print i

這樣做不僅快而且不會(huì)給內(nèi)存帶來壓力,因?yàn)槲覀兯枰闹刀际莿?dòng)態(tài)生成的而不是將他們存儲(chǔ)在一個(gè)列表中。更概括的說如果現(xiàn)在我們?cè)谏厦娴睦又惺褂?code>yield便可獲得了一個(gè)協(xié)程。協(xié)程會(huì)消費(fèi)掉發(fā)送給它的值。Python實(shí)現(xiàn)的grep就是個(gè)很好的例子:

    def grep(pattern):
        print("Searching for", pattern)
        while True:
            line = (yield)
            if pattern in line:
                print(line) 

等等!yield返回了什么?啊哈,我們已經(jīng)把它變成了一個(gè)協(xié)程。它將不再包含任何初始值,相反要從外部傳值給它。我們可以通過send()方法向它傳值。這有個(gè)例子:

    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!

發(fā)送的值會(huì)被yield接收。我們?yōu)槭裁匆\(yùn)行next()方法呢?這樣做正是為了啟動(dòng)一個(gè)協(xié)程。就像協(xié)程中包含的生成器并不是立刻執(zhí)行,而是通過next()方法來響應(yīng)send()方法。因此,你必須通過next()方法來執(zhí)行yield表達(dá)式。

我們可以通過調(diào)用close()方法來關(guān)閉一個(gè)協(xié)程。像這樣:

    search = grep('coroutine')
    search.close()

更多協(xié)程相關(guān)知識(shí)的學(xué)習(xí)大家可以參考David Beazley的這份精彩演講。