异步IO
对于异步IO阻塞非阻塞,我觉得下面的例子更简单易懂。
老张爱喝茶,废话不说,煮开水。 出场人物:老张,水壶两把(普通水壶,简称水壶;会响的水壶,简称响水壶)。 1 老张把水壶放到火上,立等水开。(同步阻塞) 老张觉得自己有点傻 2 老张把水壶放到火上,去客厅看电视,时不时去厨房看看水开没有。(同步非阻塞) 老张还是觉得自己有点傻,于是变高端了,买了把会响笛的那种水壶。水开之后,能大声发出嘀~~~~的噪音。 3 老张把响水壶放到火上,立等水开。(异步阻塞) 老张觉得这样傻等意义不大 4 老张把响水壶放到火上,去客厅看电视,水壶响之前不再去看它了,响了再去拿壶。(异步非阻塞) 老张觉得自己聪明了。
所谓同步异步,只是对于水壶而言。 普通水壶,同步;响水壶,异步。 虽然都能干活,但响水壶可以在自己完工之后,提示老张水开了。这是普通水壶所不能及的。 同步只能让调用者去轮询自己(情况2中),造成老张效率的低下。
所谓阻塞非阻塞,仅仅对于老张而言。 立等的老张,阻塞;看电视的老张,非阻塞。 情况1和情况3中老张就是阻塞的,媳妇喊他都不知道。虽然3中响水壶是异步的,可对于立等的老张没有太大的意义。所以一般异步是配合非阻塞使用的,这样才能发挥异步的效用。
生成器进阶
关于yield和yield from
def generator_1(titles):
yield titles
def generator_2(titles):
yield from titles
titles = ['Python','JavaScript','C']
for title in generator_1(titles):
print('生成器1: %s' % title)
for title in generator_2(titles):
print('生成器2: %s' % title)
本例中第二个生成器使用了yield from,等同于使用
for title in titles:
yiled title
yield from还能让我们省去很多异常处理的操作,因为其内部已经实现了大部分的异常处理。
@asyncio.coroutine装饰器是协程函数的标志,在一个任务函数前加这个装饰器,并在函数中使用yield from即可完成协程操作。
生成器不但可以产出值,还可以接收值。
send()
启动生成器的方式有两种,一种是next(),另外一种是send()
在调用send()发送非None值之前,我们必须启动一次生成器,方式有两种
- 使用gen.send(None)
- 使用一个next()
send()方法可以传递值进入生成器内部,同时还可以重启生成器执行到下一步.