浅析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的实现示例#!/usr/bin/python#encodi
- 1.开发jQuery 插件的基本格式 (function ($) { $.extend($.fn, { }) })(jQuery) 2.开发
- Django crontab定时任务安装pip install django-crontab配置在settings.py中 INSTALLE
- 在项目开发中,异常处理是不可或缺的。异常处理帮助人们debug,通过更加丰富的信息,让人们更容易找到bug的所在。异常处理还可以提高程序的容
- 动态加载JavaScript文件和CSS资源为Web前端开发提供了巨大的灵活性,同时也实现了lazy load和按需加载,相比XMLHttp
- pycharm确实要比Spyder功能全面,但是anaconda的环境配置实在太方便了,所以想take both,其实很简单,利用pycha
- 本文主要的目的是通过一个简单的例子,展示`get_absolute_url`的用法,抛砖引玉,理解实例方法的本质,能够在不同的业务场景下,灵
- 我就废话不多说了,大家还是直接看代码吧~#aaa.py#version 3.5import os #这句是没用了,不知道为什么markdow
- 字典求和edge_weights = defaultdict(lambda: defaultdict(float))for idx,node
- vue Print.js打印页面样式不出现解决方案加上这句就好了!完美!vue-print-nb打印问题总结1、表格的列缺失(element
- 一、什么是变量在读这篇文章前,我们需要搞懂到底是什么变量,其实一句话就能概括:变量是一个可以保存任何数据类型值的命名占位符。本篇文章将会介绍
- 什么是异常?异常是一个事件,其中一个程序,破坏程序的指令的正常流的执行过程中而发生的。一般情况下,当一个Python脚本遇到一些情况不能处理
- Python字典的创建方法一:>>> blank_dict = {}>>> product_dict =
- 本文实例为大家分享了python自动化生成IOS的图标,供大家参考,具体内容如下每次上架之前都要生成十几个图片感觉无聊麻烦,考虑使用脚本处理
- 正在看的ORACLE教程是:MySQL安全性指南(3)(转)。 &nb
- 记得导入包,其他按键可类比def keyPressEvent(self, event): if event.key() == Q
- eval 跟json.loads 是不一样的函数,是有实现不一样功能的地方,但是在某些地方它们两个函数的功能是一样的,在这个时候如果对执行效
- 本文为大家分享了python实现彩票系统的具体代码,供大家参考,具体内容如下功能:1、注册 2、登录 3、充钱&nb
- 通过创建变量var a = 'something' + '  
- 这本入门手册是否合适你?我只想告诉你我非常喜欢这本书。我对Microsoft Access的经验足以让我跳过这本傻瓜系列教材,但是实际情况是