浅析python协程相关概念

作者:laozhang 时间:2021-06-28 07:43:16 

这篇文章是读者朋友的python协程的学习经验之谈,以下是全部内容:

协程的历史说来话长,要从生成器开始讲起。

如果你看过我之前的文章python奇遇记:迭代器和生成器 ,对生成器的概念应该很了解。生成器节省内存,用的时候才生成结果。

 


# 生成器表达式
a = (x*x for x in range(10))
# next生成值
next(a()) # 输出0
next(a()) # 输出1
next(a()) # 输出4

与生成器产出数据不同的是,协程在产出数据的同时还可以接收数据,具体来说就是把yield 放在了表达式的右边。我们可以使用.send() 把数据发送给协程函数。


def writer():
 print('-> coroutine started')
 for i in range(8):
   w = yield
   print(i+w)

w = writer()
# 本质还是生成器
>>> w
<generator object writer at 0x000002595BC57468>
# 首先要用next()把协程激活
>>> next(w)
-> coroutine started
# 发送数据
>>> w.send(1)
1
# send到第八次之后会抛出异常
# 因为协程已经结束了
---------------------------------------------------------------------------
StopIteration               Traceback (most recent call last)

第一步必须使用next() 激活协程函数,这样才能在下一步使用.send() 发送数据。

可以看到,在第8次接收完数据之后,会产生结束的异常,因为程序流程结束了,这是正常现象。加个异常处理即可。如果需要在两个协程间传递数据呢?


def writer():
 while True:
   w = yield
   print('>>', w)

def writer_wrapper(coro):
 # 激活
 next(coro)
 while True:
   # 异常处理
   try:
     x = yield
     # 发送数据给writer
     coro.send(x)
   except StopIteration:
     pass
w = writer()
wrap = writer_wrapper(w)
# 激活
next(wrap)
for i in range(4):
 wrap.send(i)
# 输出
>> 0
>> 1
>> 2
>> 3

上面的代码中,数据首先传递到writer_wrapper,之后再传递到writer 。

data——>writer_wrapper——>writer

可以这么写,不过,又要预先激活,又要加异常,看起来有点麻烦啊。yield from 的出现可以解决这个问题,同样是传递数据:


def writer():
 while True:
   w = yield
   print('>>', w)
def writer_wrapper2(coro):
 yield from coro

一行代码解决问题。

总之,yield from相当于提供了一个通道,使得数据可以在协程之间流转 。writer_wrapper2 中使用yield from coro时,coro此时获得控制权,在我们.send() 数据时,writer_wrapper2 被阻塞,直到writer 打印出结果。

在这个阶段,协程本质上还是由生成器构成的。

即使我们使用yield from 简化了流程,协程和生成器的知识理解起来还是有点懵逼,而且yield from 用在异步编程中有诸多不顺(asyncio以前就是用yield from),于是在3.5版本的python中,弃用了yield from ,新加入了两个关键字async 和await ,同时协程不再是生成器类型,而是原生的协程类型。

现在我们定义一个协程要像下面这样:


async def func():
 await 'some code'

不用于异步的协程该怎么用,我还不知道。所以,协程的介绍到这里就结束啦。感谢你对脚本之家的支持。

标签:python,协程
0
投稿

猜你喜欢

  • 网马解密大讲堂——网马解密初级篇

    2009-09-16 14:45:00
  • Python如何把Spark数据写入ElasticSearch

    2021-06-28 20:07:46
  • Python Socket编程入门教程

    2022-03-08 01:08:49
  • 清理你的CSS

    2009-10-06 15:11:00
  • Python用61行代码实现图片像素化的示例代码

    2021-08-27 13:10:55
  • Python3爬虫mitmproxy的安装步骤

    2022-08-26 12:23:26
  • PHP入门教程之面向对象的特性分析(继承,多态,接口,抽象类,抽象方法等)

    2023-11-18 08:17:16
  • Python中最快的循环姿势实例详解

    2022-12-16 01:30:50
  • BigPipe:高性能的"流水线技术"网页

    2010-11-02 12:47:00
  • 利用Python绘制多种风玫瑰图

    2023-05-05 03:41:50
  • 单击按钮将内容复制到剪贴板

    2008-08-22 13:08:00
  • Python self用法详解

    2021-08-24 04:26:41
  • MS SQL Server中的CONVERT日期格式化大全

    2010-08-07 11:31:00
  • 如何快速通过XSL转换XML文件

    2023-07-02 21:22:40
  • 详解配置Django的Celery异步之路踩坑

    2022-11-25 22:06:45
  • 分享我们的select控件设计过程

    2009-06-16 18:04:00
  • Python实现读取并保存文件的类

    2022-01-12 06:45:35
  • 解决在keras中使用model.save()函数保存模型失败的问题

    2021-09-12 17:32:41
  • Python三种遍历文件目录的方法实例代码

    2023-06-26 05:24:50
  • python绘制发散型柱状图+误差阴影时间序列图+双坐标系时间序列图+绘制金字塔图

    2023-09-19 17:50:36
  • asp之家 网络编程 m.aspxhome.com