# 协程
# 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+
- 上下文管理器
- 基于类的实现
- 处理异常
- 基于生成器的实现