python---await/async关键字
推文:玩转 Python 3.5 的 await/async
首先看正常的两个函数之间的执行
def func1():
print("func1 start")
print("func1 end") def func2():
print("func2 start")
print("func2 a")
print("func2 b")
print("func2 c")
print("func2 end") func1()
func2()
func1 start
func1 end
func2 start
func2 a
func2 b
func2 c
func2 end
无法实现两个函数之间的交互。
将这两个函数设置为协程,加上async关键字
async def func1():
print("func1 start")
print("func1 end") async def func2():
print("func2 start")
print("func2 a")
print("func2 b")
print("func2 c")
print("func2 end") f1 = func1()
f2 = func2()
print(f1,f2)
<coroutine object func1 at 0x000000000084C468> <coroutine object func2 at 0x000000000084CD58>
sys:: RuntimeWarning: coroutine 'func1' was never awaited #警告信息
sys:: RuntimeWarning: coroutine 'func2' was never awaited #上面不再只是函数,而变为协程对象,协程在调用时,不会被执行
那么,为什么要有一个协程对象?代码到底如何执行?
关键之处是协程确实是与 Python 的生成器非常相似,也都有一个 send 方法。我们可以通过调用 send 方法来启动一个协程的执行。
f1 = func1()
f2 = func2() try:
print('f1.send')
f1.send(None)
except StopIteration as e: #这里也是需要去捕获StopIteration方法
pass try:
print('f1.send')
f2.send(None)
except StopIteration as e:
pass
StopIteration 异常是一种标记生成器(或者像这里的协程)执行结束的机制。虽然这是一个异常,但是确实是我们期望的!我们可以用适当的 try-catch 代码将其包起来,这样就可以避免错误提示。
上面虽然将两个协程都执行起来了,但是并没有做到交替执行,和最初的并无不同,而且似乎更加麻烦
协程与线程相似的地方是多个线程之间也可以交替执行,不过与线程不同之处在于协程之间的切换是显式的,而线程是隐式的(大多数情况下是更好的方式)。所以我们需要加入显式切换的代码。
我们可能根据生成器或者协程gevent的使用,而去推测这种方案:(是错误的)
async def func1():
print("func1 start")
yield
print("func1 end")
不允许在本地协程函数中使用yield,但是作为替换,我们可以使用到await 表达式来暂停协程的执行。注意await _something_,这里的_something_代表的是基于生成器的协程对象,或者一个特殊的类似 Future 的对象。
下面使用一个基于生成器的协程函数
import types @types.coroutine
def switch():
yield async def func1():
print("func1 start")
await switch()
print("func1 end") async def func2():
print("func2 start")
print("func2 a")
print("func2 b")
print("func2 c")
print("func2 end") f1 = func1()
f2 = func2() try:
f1.send(None)
except StopIteration as e:
pass try:
f2.send(None)
except StopIteration as e:
pass
func1 start
func2 start
func2 a
func2 b
func2 c
func2 end
注意上面的输出:只是执行了函数的转换,有func1转换到func2,并没有再回到func1中。那么如何让其在回到func1中呢。
我们可以基于生成器一样再次对func1协程使用send方法,再次激活yield向下执行
import types @types.coroutine
def switch():
yield async def func1():
print("func1 start")
await switch()
print("func1 end") async def func2():
print("func2 start")
print("func2 a")
print("func2 b")
print("func2 c")
print("func2 end") f1 = func1()
f2 = func2() try:
f1.send(None)
except StopIteration as e:
pass try:
f2.send(None)
except StopIteration as e:
pass try:
f1.send(None)
except StopIteration as e:
pass
func1 start
func2 start
func2 a
func2 b
func2 c
func2 end
func1 end
上面可以实现两个函数交替一次,回到原协程中。但是我们更加希望连续不断的调用send驱动不同的协程去执行,直到send抛出 StopIteration 异常。
为此我们
新建一个函数,这个函数传入一个协程列表,函数执行这些协程直到全部结束。我们现在要做的就是调用这个函数。
import types @types.coroutine
def switch():
yield async def func1():
print("func1 start")
await switch()
print("func1 e")
await switch()
print("func1 end") async def func2():
print("func2 start")
print("func2 a")
await switch()
print("func2 b")
print("func2 c")
await switch()
print("func2 end") def run(task_list):
coro_list = list(task_list) while coro_list:
for coro in list(coro_list):
try:
coro.send(None)
except StopIteration:
coro_list.remove(coro) f1 = func1()
f2 = func2() run([f1,f2])
python---await/async关键字的更多相关文章
- 对于Linq关键字和await,async异步关键字的扩展使用
最近在看neuecc大佬写的一些库:https://neuecc.medium.com/,其中对await,async以及linq一些关键字实现了自定义化使用, 使其不需要引用对应命名空间,不需要多线 ...
- javascript异步编程的前世今生,从onclick到await/async
javascript与异步编程 为了避免资源管理等复杂性的问题, javascript被设计为单线程的语言,即使有了html5 worker,也不能直接访问dom. javascript 设计之初是为 ...
- 5分种让你了解javascript异步编程的前世今生,从onclick到await/async
javascript与异步编程 为了避免资源管理等复杂性的问题,javascript被设计为单线程的语言,即使有了html5 worker,也不能直接访问dom. javascript 设计之初是 ...
- await和async关键字来写异步程序
await和async关键字出现于.Net5.0,方便写异步程序. 例子: public class MyClass { public MyClass() { DisplayValue(); //这里 ...
- C#异步编程のawait和async关键字来写异步程序
一.await和async关键字 .Net平台不断推出了新的异步编程模型,在.net4.5中加入了关键字await和async,顾名思义,await是指方法执行可等待,即可挂起直到有结果(不是必须立即 ...
- C#~异步编程续~.net4.5主推的await&async应用
返回目录 之前写过两篇关于异步编程的文章,详细可以进入C#~异步编程和C#~异步编程在项目中的使用 .net的各个版本都有自己主推的技术,像.NET1.1中的委托,.NET2.0中的泛型,.NET3. ...
- C#~异步编程续~.net4.5主推的await&async应用(转)
之前写过两篇关于异步编程的文章,详细可以进入C#~异步编程和C#~异步编程在项目中的使用 .net的各个版本都有自己主推的技术,像.NET1.1中的委托,.NET2.0中的泛型,.NET3.0中的Li ...
- Silverlight项目笔记1:UI控件与布局、MVVM、数据绑定、await/async、Linq查询、WCF RIA Services、序列化、委托与事件
最近从技术支持转到开发岗,做Silverlight部分的开发,用的Prism+MVVM,框架由同事搭好,目前做的主要是功能实现,用到了一些东西,侧重于如何使用,总结如下 1.UI控件与布局 常用的主要 ...
- 如何避免 await/async 地狱
原文地址:How to escape async/await hell 译文出自:夜色镇歌的个人博客 async/await 把我们从回调地狱中解救了出来,但是如果滥用就会掉进 async/await ...
随机推荐
- BETA-2
前言 我们居然又冲刺了·二 团队代码管理github 站立会议 队名:PMS 530雨勤(组长) 过去两天完成了哪些任务 了解OpenCV下的视频参数及其调用方法 初步编码 接下来的计划 文档工作 速 ...
- 蜗牛慢慢爬 LeetCode 15. 3Sum [Difficulty: Medium]
题目 Given an array S of n integers, are there elements a, b, c in S such that a + b + c = 0? Find all ...
- Scrum Meeting Beta - 7
Scrum Meeting Beta - 7 NewTeam 2017/12/6 地点:新主楼F座二楼 任务反馈 团队成员 完成任务 计划任务 安万贺 修复离线状态下启动时的bugIssue #150 ...
- Log4j读取配置文件并使用
/** 设置配置路径从环境变量读取 * PropertyConfigurator类加载.properties文件的配置 * DOMConfigurator加载.xml文件的配置 ...
- 【bzoj5166】[HAOI2014]遥感监测 贪心
题目描述 给出平面上 $n$ 个圆,在x轴上选出尽可能少的点,使得每个圆中至少有一个点.求这个最小点数. 输入 第1行: N R 分别表示激光点的个数和射电望远镜能检测到的半径 第2~N+1行: Xi ...
- 【大数据】Spark基础解析
第1章 Spark概述 1.1 什么是Spark 1.2 Spark内置模块 Spark Core:实现了Spark的基本功能,包含任务调度.内存管理.错误恢复.与存储系统交互等模块.Spark Co ...
- A Shade of Moonlight CodeForces - 989D
看这位博主的吧:https://blog.csdn.net/Steaunk/article/details/80660954
- 【刷题】BZOJ 2588 Spoj 10628. Count on a tree
Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始 ...
- 【刷题】BZOJ 4031 [HEOI2015]小Z的房间
Description 你突然有了一个大房子,房子里面有一些房间.事实上,你的房子可以看做是一个包含n*m个格子的格状矩形,每个格子是一个房间或者是一个柱子.在一开始的时候,相邻的格子之间都有墙隔着. ...
- codevs2464超级麻将
题目链接http://codevs.cn/problem/2464/ 题目描述 Description 很多人都知道玩麻将,当然也有人不知道,呵呵,不要紧,我在这里简要地介绍一下麻将规则: 普通麻将有 ...