asyncio异步IO--同步原语
asyncio同步原语与线程(threading)模块同步原语基本类似,但有两点重要区别:
- asyncio同步原语
非线程安全,因此不应被用作系统线程同步(可以使用threading代替); - asyncio同步原语不允许使用
timeout参数;可以使用asyncio.wait_for()方法执行有超时设置的操作。
asyncio有以下5个基本的同步原语:
- Lock
- Event
- Condition
- Semaphore
- BoundedSemaphore
Lock
- class asyncio.Lock(*,loop=None)
- 为asyncio任务提供一个
互斥锁。非线程安全。 - asyncio锁可以用来保证对共享资源的独占访问。
- asyncio锁的首选用法是同
async with语句一起使用:lock = asyncio.Lock()
# ... later
async with lock:
# 访问共享资源
此代码段和以下代码是等价的:
lock = asyncio.Lock() # ... later
await lock.acquire()
try:
# 访问共享资源
finally:
lock.release()
- coroutine acquire()
- 获取asyncio同步锁。
- 该方法等待
锁的状态变为unlocked,之后设置其为locked,并返回True。
- release()
- 释放asyncio同步锁。
- 如果
锁的状态是locked,则将其重置为unlocked并返回。 - 如果
锁的状态是unlocked,会引发RuntimeError异常。
- locked()
- 如果
锁的状态是locked,则返回True。
- 如果
- 为asyncio任务提供一个
Event
- class asyncio.Event(*,loop=None)
- 事件对象,非线程安全。
- 用于向asyncio任务通知某些事件已发生。
- 事件对象用于管理
内部标志。此标志可以通过set()方法设置为True,或通过clear()方法复位为False。wait()方法在该标志设置为True前一直保持阻塞。初始状态下,该标志为False。 - 例如:
async def waiter(event):
print('waiting for it ...')
await event.wait()
print('... got it!') async def main():
# Create an Event object.
event = asyncio.Event() # Spawn a Task to wait until 'event' is set.
waiter_task = asyncio.create_task(waiter(event)) # Sleep for 1 second and set the event.
await asyncio.sleep(1)
event.set() # Wait until the waiter task is finished.
await waiter_task asyncio.run(main())
- coroutine wait()
- 等待事件
内部标志被设置为True。 - 如果事件的内部
内部标志已设置,则立即返回True。否则,一直阻塞,直到另外的任务调用set()。
- 等待事件
- set()
- 设置事件
内部标志为True。 - 所有等待事件的任务将会立即被触发。
- 设置事件
- clear()
- 清除事件
内部标志(即设置为False)。 - 等待事件的任务将会阻塞,直到
set()方法被再次调用。
- 清除事件
- is_set()
- 如果事件
内部标志被设置为True,则返回True。
- 如果事件
Condition
- class asyncio.Condition(lock=None,*,loop=None)
- 条件对象,非线程安全。
- 异步条件原语用于在某些事件发生后,获得共享资源的独占访问权限。
- 本质上,条件对象结合了事件和锁的功能。可以让多个Condition对象共享一个Lock,这允许在对共享资源的特定状态感兴趣的不同任务之间协调对共享资源的独占访问。
- 可选参数
lock必须为Lock对象或None。如果为None,会自动创建一个Lock对象。 - 使用条件对象的首选方法是
async with方式:cond = asyncio.Condition() # ... later
async with cond:
await cond.wait()
等价于:
cond = asyncio.Condition() # ... later
await lock.acquire()
try:
await cond.wait()
finally:
lock.release()
- coroutine acquire()
- 获取底层锁。
- 该方法一直等待,直到底层锁处于未锁定状态,然后设置其为锁定状态,并且返回
True。
- notify(n=1)
- 唤醒至多
n个等待条件的任务。如果没有正在等待的任务,则该方法无操作。 - 在调用该方法之前,必须先调用
acquire()获取锁,并在调用该方法之后释放锁。如果在锁为锁定的情况下调用此方法,会引发RuntimeError异常。
- 唤醒至多
- locked()
- 如果底层锁已获取,则返回
True。
- 如果底层锁已获取,则返回
- notify_all()
- 唤醒所有正在等待该条件的任务。
- 该方法与
notify()类似,区别只在它会唤醒所有正在等待的任务。 - 在调用该方法前,必须首先获取底层锁,并在执行完该方法后释放锁。如果在底层锁未锁定的情况下执行该方法,会引发
RuntimeError异常。
- release()
- 释放底层锁。
- 在未锁定的锁上调用时,会引发
RuntimeError异常。
- coroutine wait()
- 等待通知。
- 如果调用此方法的任务没有获取到锁,则引发
RuntimeError异常。 - 此方法释放底层锁,然后保持阻塞,直至被
notify()或notify_all()唤醒。被唤醒之后,条件对象重新申请锁,该方法返回True。
- coroutine wait_for(predicate)
- 等待
predicate变为True。 predicate必须可调用,它的执行结果会被解释为布尔值,并作为最终结果返回。
- 等待
Semaphore
- class asyncio.Semaphore(value=1,*,loop=None)
- 信号量(Semaphore)对象。非线程安全。
- 信号量用于管理一个内部计数器,此计数器逢
acquire()递减,逢release()递增。计数器的值不能小于0,如果acquire()被调用时计数器为0,则阻塞,直到某一任务调用release()。 value为可选参数,用于设定内部计数器的初始值。如果给定的值小于0,则引发ValueError异常。- 使用
信号量的最佳方法是async with声明:sem = asyncio.Semaphore(10) # ... later
async with sem:
# work with shared resource
等价于:
sem = asyncio.Semaphore(10) # ... later
await sem.acquire()
try:
# work with shared resource
finally:
sem.release()
- coroutine acquire()
- 申请一个信号量
- 如果内部计数器的值大于0,则减1并立即返回
True。如果内部计数器的值为0,则等待release()被调用,然后返回True。
- locked()
- 如果信号量不能被立即申请,则返回
True。
- 如果信号量不能被立即申请,则返回
- release()
- 释放信号量,内部计数器加1。
- 与
BoundedSemaphore不同,Semaphore允许release()的调用次数大于acquire()的调用次数。
BoundedSemaphore
- class asyncio.BoundedSemaphore(value=1,*,loop=None)
- 有界信号量,非线程安全。
- 有界信号量是一种特殊的信号量——如果
release()后内部计数器的值大于初始值,则引发ValueError异常。
从python3.7开始:通过await lock、yield from lock或通过声明with await lock、with(yield from lock)获取锁的用法被废弃。可使用async with lock代替。
asyncio异步IO--同步原语的更多相关文章
- 【Python学习之九】asyncio—异步IO
asyncio 这是python3.4引入的标准库,直接内置对异步IO的支持.asyncio的编程模型就是一个消息循环.从asyncio模块中直接获取一个EventLoop的引用,然后把需要执行的协程 ...
- asyncio异步IO——Streams详解
前言 本文翻译自python3.7官方文档--asyncio-stream,译者马鸣谦,邮箱 1612557569@qq.com.转载请注明出处. 数据流(Streams) 数据流(Streams)是 ...
- 5种IO模型、阻塞IO和非阻塞IO、同步IO和异步IO
POSIX 同步IO.异步IO.阻塞IO.非阻塞IO,这几个词常见于各种各样的与网络相关的文章之中,往往不同上下文中它们的意思是不一样的,以致于我在很长一段时间对此感到困惑,所以想写一篇文章整理一下. ...
- 爬虫之多线程 多进程 自定义异步IO框架
什么是进程? 进程是程序运行的实例,是系统进行资源分配和调度的一个独立单位,它包括独立的地址空间,资源以及1个或多个线程. 什么是线程? 线程可以看成是轻量级的进程,是CPU调度和分派的基本单位. 进 ...
- 2020.11.2 异步IO 协程
异步IO 同步IO在一个线程中,CPU执行代码的速度极快,然而,一旦遇到IO操作,如读写文件.发送网络数据时,就需要等待IO操作完成,才能继续进行下一步操作. 在IO操作的过程中,当前线程被挂起,而其 ...
- 【译】深入理解python3.4中Asyncio库与Node.js的异步IO机制
转载自http://xidui.github.io/2015/10/29/%E6%B7%B1%E5%85%A5%E7%90%86%E8%A7%A3python3-4-Asyncio%E5%BA%93% ...
- Python并发编程之初识异步IO框架:asyncio 上篇(九)
大家好,并发编程 进入第九篇. 通过前两节的铺垫(关于协程的使用),今天我们终于可以来介绍我们整个系列的重点 -- asyncio. asyncio是Python 3.4版本引入的标准库,直接内置了对 ...
- (转)Python黑魔法 --- 异步IO( asyncio) 协程
转自:http://www.jianshu.com/p/b5e347b3a17c?from=timeline Python黑魔法 --- 异步IO( asyncio) 协程 作者 人世间 关注 201 ...
- python中同步、多线程、异步IO、多线程对IO密集型的影响
目录 1.常见并发类型 2.同步版本 3.多线程 4.异步IO 5.多进程 6.总结 1.常见并发类型 I/ O密集型: 蓝色框表示程序执行工作的时间,红色框表示等待I/O操作完成的时间.此图没有按比 ...
随机推荐
- IOS微信点击input弹出输入法,关闭后页面留白解决方案
场景:IOS用微信点击input框弹出输入法后 不管你是输入信息,还是不输入直接点完成关闭输入法,都会导致页面被挤上去后产生留白,从而改变页面布局 解决方法: 给input添加 ...
- [Swift]LeetCode1007. 行相等的最少多米诺旋转 | Minimum Domino Rotations For Equal Row
In a row of dominoes, A[i] and B[i] represent the top and bottom halves of the i-th domino. (A domi ...
- Linux 中改变主机名的 4 种方法
今天,我们将向你展示使用不同的方法来修改主机名.你可以从中选取最适合你的方法. 使用 systemd 的系统自带一个名为 hostnamectl 的好用工具,它可以使我们能够轻易地管理系统的主机名. ...
- dataframe去除null、NaN和空字符串
去除null.NaN 去除 dataframe 中的 null . NaN 有方法 drop ,用 dataframe.na 找出带有 null. NaN 的行,用 drop 删除行: import ...
- python3安装sklearn机器学习库
安装sklearn需要的库请全部在万能仓库下载: http://www.lfd.uci.edu/~gohlke/pythonlibs/#scipy http://www.lfd.uci.edu/~go ...
- Ocelot统一权限验证
Ocelot作为网关,可以用来作统一验证,接上一篇博客,我们继续 前一篇,我们创建了OcelotGateway网关项目,DemoAAPI项目,DemoBAPI项目,为了验证用户并分发Token,现在还 ...
- Jenkins 无法捕获构建脚本错误问题
Jenkins 版本 2.121.1 编写构建脚本执行,发现脚本执行出错,不会中断构建过程,导致最后展现的构建结果是错误的. 原因:构建脚本头部加入 #!/bin/bash ,jenkins会将脚本放 ...
- Owin学习笔记(一) Owin的前生今世
ASP.NET框架至今为止已经存在了数十年了,大量的网站使用ASP.NET框架进行开发.随着网站应用开发技术的进步, 许多网站应用开发框架有了新的流行趋势 轻量化 模块化 可移植 ASP.NET框架 ...
- Linux基础知识第二讲,文件目录命令使用
目录 一丶Linux终端使用技巧. 1.自动补全 Tab技巧. 2.使用输入过的命令 二丶Linux 目录知识 1.linux目录的特点 2.ls 隐藏文件的查看 3.ls 常用选项 4.通配符的配合 ...
- RDIFramework.NET ━ .NET快速信息化系统开发框架 V3.2->Web版本新增新的角色授权管理界面效率更高、更规范
角色授权管理模块主要是对角色的相应权限进行集中设置.在角色权限管理模块中,管理员可以添加或移除指定角色所包含的用户.可以分配或授予指定角色的模块(菜单)的访问权限.可以收回或分配指定角色的操作(功能) ...