当前位置: 移动技术网 > IT编程>脚本编程>Python > python 使用事件对象asyncio.Event来同步协程的操作

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

2020年05月12日  | 移动技术网IT编程  | 我要评论

www.4399.con,五股麻花辫的编法,海贼王530

事件对象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来同步协程的操作就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持移动技术网。

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网