浅析python协程相关概念


Posted in Python onJanuary 20, 2018

这篇文章是读者朋友的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 相关文章推荐
用python实现的可以拷贝或剪切一个文件列表中的所有文件
Apr 30 Python
在Python中使用Mako模版库的简单教程
Apr 08 Python
pygame学习笔记(6):完成一个简单的游戏
Apr 15 Python
Python中方法链的使用方法
Feb 23 Python
Python的Socket编程过程中实现UDP端口复用的实例分享
Mar 19 Python
解决python3 urllib中urlopen报错的问题
Mar 25 Python
Python错误: SyntaxError: Non-ASCII character解决办法
Jun 08 Python
python爬虫爬取某站上海租房图片
Feb 04 Python
python Celery定时任务的示例
Mar 13 Python
python 基本数据类型占用内存空间大小的实例
Jun 12 Python
python中requests模拟登录的三种方式(携带cookie/session进行请求网站)
Nov 17 Python
如何使用Tkinter进行窗口的管理与设置
Jun 30 Python
Python实现可自定义大小的截屏功能
Jan 20 #Python
python+matplotlib绘制饼图散点图实例代码
Jan 20 #Python
Python+matplotlib绘制不同大小和颜色散点图实例
Jan 19 #Python
Python搭建FTP服务器的方法示例
Jan 19 #Python
python re模块findall()函数实例解析
Jan 19 #Python
Python enumerate索引迭代代码解析
Jan 19 #Python
5个很好的Python面试题问题答案及分析
Jan 19 #Python
You might like
Netflix将与CLAMP、乙一以及冲方丁等6名知名制作人合伙展开原创动画计划!
2020/03/06 日漫
加速XP搜索功能堪比vista
2007/03/22 PHP
PHP设计模式之装饰者模式
2012/02/29 PHP
ThinkPHP采用原生query实现关联查询left join实例
2014/12/02 PHP
php中使用GD库做验证码
2016/03/31 PHP
PHP简单日历实现方法
2016/07/20 PHP
Sample script that deletes a SQL Server database
2007/06/16 Javascript
Pro JavaScript Techniques学习笔记
2010/12/28 Javascript
JavaScript中将一个值转换为字符串的方法分析[译]
2012/09/21 Javascript
js获得页面的高度和宽度的方法
2014/02/23 Javascript
谈谈impress.js初步理解
2015/09/09 Javascript
javascript中的previousSibling和nextSibling的正确用法
2015/09/16 Javascript
JavaScript+html5 canvas绘制的圆弧荡秋千效果完整实例
2016/01/26 Javascript
bootstrap组件之按钮式下拉菜单小结
2017/01/19 Javascript
Angular2中如何使用ngx-translate进行国际化
2017/05/21 Javascript
详解Vue2 SSR 缓存 Api 数据
2017/11/20 Javascript
React中如何引入Angular组件详解
2018/08/09 Javascript
vue安装和使用scss及sass与scss的区别详解
2018/10/15 Javascript
详解Vue.js 响应接口
2020/07/04 Javascript
python基础教程之基本内置数据类型介绍
2014/02/20 Python
用Python从零实现贝叶斯分类器的机器学习的教程
2015/03/31 Python
Python计算斗牛游戏概率算法实例分析
2017/09/26 Python
python opencv旋转图像(保持图像不被裁减)
2018/07/26 Python
对Python2与Python3中__bool__方法的差异详解
2018/11/01 Python
Python3中urlencode和urldecode的用法详解
2019/07/23 Python
Django实现CAS+OAuth2的方法示例
2019/10/30 Python
Python中zip()函数的解释和可视化(实例详解)
2020/02/16 Python
Django 解决上传文件时,request.FILES为空的问题
2020/05/20 Python
Python设计密码强度校验程序
2020/07/30 Python
python实现计算图形面积
2021/02/22 Python
python推导式的使用方法实例
2021/02/28 Python
Eyeko美国:屡获殊荣的睫毛膏、眼线笔和眉妆
2018/07/05 全球购物
国防教育标语
2014/10/08 职场文书
公司财务经理岗位职责
2015/04/08 职场文书
解决ObjectMapper.convertValue() 遇到的一些问题
2021/06/30 Java/Android
Go归并排序算法的实现方法
2022/04/06 Golang