python 使用事件对象asyncio.Event来同步协程的操作


Posted in Python onMay 04, 2020

事件对象asyncio.Event是基于threading.Event来实现的。

事件可以一个信号触发多个协程同步工作,

例子如下:

import asyncio
import functools
 
def set_event(event):
  print('setting event in callback')
  event.set()
 
async def coro1(event):
  print('coro1 waiting for event')
  await event.wait()
  print('coro1 triggered')
 
async def coro2(event):
  print('coro2 waiting for event')
  await event.wait()
  print('coro2 triggered')
 
async def main(loop):
  # Create a shared event
  event = asyncio.Event()
  print('event start state: {}'.format(event.is_set()))
 
  loop.call_later(
    0.1, functools.partial(set_event, event)
  )
 
  await asyncio.wait([coro1(event), coro2(event)])
  print('event end state: {}'.format(event.is_set()))
 
event_loop = asyncio.get_event_loop()
try:
  event_loop.run_until_complete(main(event_loop))
finally:
  event_loop.close()

输出如下:

event start state: False
coro2 waiting for event
coro1 waiting for event
setting event in callback
coro2 triggered
coro1 triggered
event end state: True

补充知识: python里使用协程来创建echo客户端

在这个例子里使用asyncio.Protocol来创建一个echo客户端,先导入库asyncio和logging。

接着定义发送的消息MESSAGES。

创建连接服务器的地址SERVER_ADDRESS,接着创建EchoClient类,它是继承asyncio.Protocol。

在这个类的构造函数里,接收两个参数messages和future,

messages是指定要发送的消息数据,future是用来通知socket接收数据完成或者服务器关闭socket的事件通知,以便事件循环知道这个协程已经完成了,就可以退出整个程序。

connection_made函数是当socket连接到服务器时调用,它就立即发送数据给服务器,数据发送完成之后发送了eof标记。

服务器收到数据和标志都回复客户端,客户端data_received函数接收数据,eof_received函数接收结束标记。

connection_lost函数收到服务器断开连接。

这行代码:

client_completed = asyncio.Future()

创建一个协程完成的触发事件。

由于event_loop.create_connection函数只能接收一个参数,需要使用functools.partial来进行多个参数包装成一个参数。

后面通过事件循环来运行协程。

import asyncio
import functools
import logging
import sys
 
MESSAGES = [
  b'This is the message. ',
  b'It will be sent ',
  b'in parts.',
]
SERVER_ADDRESS = ('localhost', 10000)
 
class EchoClient(asyncio.Protocol):
 
  def __init__(self, messages, future):
    super().__init__()
    self.messages = messages
    self.log = logging.getLogger('EchoClient')
    self.f = future
 
  def connection_made(self, transport):
    self.transport = transport
    self.address = transport.get_extra_info('peername')
    self.log.debug(
      'connecting to {} port {}'.format(*self.address)
    )
    # This could be transport.writelines() except that
    # would make it harder to show each part of the message
    # being sent.
    for msg in self.messages:
      transport.write(msg)
      self.log.debug('sending {!r}'.format(msg))
    if transport.can_write_eof():
      transport.write_eof()
 
  def data_received(self, data):
    self.log.debug('received {!r}'.format(data))
 
  def eof_received(self):
    self.log.debug('received EOF')
    self.transport.close()
    if not self.f.done():
      self.f.set_result(True)
 
  def connection_lost(self, exc):
    self.log.debug('server closed connection')
    self.transport.close()
    if not self.f.done():
      self.f.set_result(True)
    super().connection_lost(exc)
 
logging.basicConfig(
  level=logging.DEBUG,
  format='%(name)s: %(message)s',
  stream=sys.stderr,
)
log = logging.getLogger('main')
 
event_loop = asyncio.get_event_loop()
 
client_completed = asyncio.Future()
 
client_factory = functools.partial(
  EchoClient,
  messages=MESSAGES,
  future=client_completed,
)
factory_coroutine = event_loop.create_connection(
  client_factory,
  *SERVER_ADDRESS,
)
 
log.debug('waiting for client to complete')
try:
  event_loop.run_until_complete(factory_coroutine)
  event_loop.run_until_complete(client_completed)
finally:
  log.debug('closing event loop')
  event_loop.close()

以上这篇python 使用事件对象asyncio.Event来同步协程的操作就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Django中实现点击图片链接强制直接下载的方法
May 14 Python
初步讲解Python中的元组概念
May 21 Python
Python实现小数转化为百分数的格式化输出方法示例
Sep 20 Python
详谈python3 numpy-loadtxt的编码问题
Apr 29 Python
python PrettyTable模块的安装与简单应用
Jan 11 Python
pytorch实现seq2seq时对loss进行mask的方式
Feb 18 Python
动态设置django的model field的默认值操作步骤
Mar 30 Python
使用python接受tgam的脑波数据实例
Apr 09 Python
parser.add_argument中的action使用
Apr 20 Python
在pycharm中创建django项目的示例代码
May 28 Python
Python定义一个函数的方法
Jun 15 Python
python 识别登录验证码图片功能的实现代码(完整代码)
Jul 03 Python
在python里使用await关键字来等另外一个协程的实例
May 04 #Python
python 异步async库的使用说明
May 04 #Python
Python插件机制实现详解
May 04 #Python
python3+selenium获取页面加载的所有静态资源文件链接操作
May 04 #Python
解决IDEA 的 plugins 搜不到任何的插件问题
May 04 #Python
python3 sleep 延时秒 毫秒实例
May 04 #Python
Python并发concurrent.futures和asyncio实例
May 04 #Python
You might like
让你的WINDOWS同时支持MYSQL4,MYSQL4.1,MYSQL5X
2006/12/06 PHP
php Xdebug 调试扩展的安装与使用.
2010/03/13 PHP
详解PHP处理字符串类似indexof的方法函数
2017/06/11 PHP
PHP设计模式之注册树模式分析
2018/01/26 PHP
实例介绍PHP删除数组中的重复元素
2019/03/03 PHP
js实现addClass,removeClass,hasClass的函数代码
2011/07/13 Javascript
stream.js 一个很小、完全独立的Javascript类库
2011/10/28 Javascript
jQuery UI Autocomplete 体验分享
2012/02/14 Javascript
javascript实例--教你实现扑克牌洗牌功能
2014/05/15 Javascript
JavaScript数组前面插入元素的方法
2015/04/06 Javascript
全面解析Bootstrap中form、navbar的使用方法
2016/05/30 Javascript
利用VUE框架,实现列表分页功能示例代码
2017/01/12 Javascript
jquery实现拖动效果(代码分享)
2017/01/25 Javascript
微信小程序 弹幕功能简单实例
2017/02/14 Javascript
详解AngularJs HTTP响应拦截器实现登陆、权限校验
2017/04/11 Javascript
vue2.0父子组件间通信的实现方法
2017/04/19 Javascript
zTree获取当前节点的下一级子节点数实例
2017/09/05 Javascript
JS使用tofixed与round处理数据四舍五入的区别
2017/10/25 Javascript
在weex中愉快的使用scss的方法步骤
2020/01/02 Javascript
[00:06]Yes,it worked!小卡尔成功穿越时空加入战场!
2019/07/20 DOTA
python使用urllib模块开发的多线程豆瓣小站mp3下载器
2014/01/16 Python
python merge、concat合并数据集的实例讲解
2018/04/12 Python
Python 实现取矩阵的部分列,保存为一个新的矩阵方法
2018/11/14 Python
使用python 打开文件并做匹配处理的实例
2019/01/02 Python
使用phonegap获取位置信息的实现方法
2017/03/31 HTML / CSS
惠普美国官方商店:HP Official Store
2016/08/28 全球购物
夜大自我鉴定
2013/10/31 职场文书
企业负责人任命书
2014/06/05 职场文书
学校学雷锋活动总结
2014/06/26 职场文书
理财学专业自荐书
2014/06/28 职场文书
单位委托书怎么写
2014/08/02 职场文书
2015欢度元旦标语口号
2014/12/09 职场文书
门卫岗位职责
2015/02/09 职场文书
考研英语辞职信
2015/05/13 职场文书
2015年卫生监督工作总结
2015/05/21 职场文书
入团申请书格式
2019/06/20 职场文书