繁体   English   中英

使用asyncio和aiohttp的多个非阻塞任务

[英]multiple nonblocking tasks using asyncio and aiohttp

我试图用asyncio和aiohttp执行几个非阻塞任务,我不认为我这样做是有效的。 我认为最好使用await而不是yield。 有人可以帮忙吗?

def_init__(self):
    self.event_loop = asyncio.get_event_loop()

def run(self):
    tasks = [
        asyncio.ensure_future(self.subscribe()),
        asyncio.ensure_future(self.getServer()),]
    self.event_loop.run_until_complete(asyncio.gather(*tasks))
    try:
       self.event_loop.run_forever()

@asyncio.coroutine
def getServer(self):
    server = yield from self.event_loop.create_server(handler, ip, port)
    return server

@asyncio.coroutine
def sunbscribe(self):
    while True:
        yield from asyncio.sleep(10)
        self.sendNotification(self.sub.recieve())

def sendNotification(msg):
    # send message as a client

我必须听一个服务器并订阅收听广播,并根据广播的消息POST到另一台服务器。

根据PEP 492

await,类似于yield from,暂停执行read_data协程,直到db.fetch等待完成并返回结果数据。

它使用实现中的yield,并使用额外的步骤验证其参数。 等待只接受一个等待的,可以是以下之一:

所以我没有在代码中看到效率问题,因为它们使用相同的实现。

但是,我确实想知道为什么要返回server但从不使用它。

我在你的代码中看到的主要设计错误是你使用两者:

self.event_loop.run_until_complete(asyncio.gather(*tasks))
try:
   self.event_loop.run_forever()

从我所看到的你只需要run_forever()

一些额外的提示:

在我使用asyncio的实现中,我通常会确保在出现错误时关闭循环,否则这会导致大量泄漏,具体取决于您的应用类型。

try:
    loop.run_until_complete(asyncio.gather(*tasks))
finally:  # close the loop no matter what or you leak FDs
    loop.close()

我也使用Uvloop而不是内置的,根据基准测试它更有效率。

import uvloop
...
    loop = uvloop.new_event_loop()
    asyncio.set_event_loop(loop)

Await不会比收益率更高效。 它可能更像pythonic,但是

async def foo():
    await some_future

@asyncio.coroutine
def foo()
    yield from some_future

大致相同。 当然,就效率而言,它们非常接近。 Await是使用与yield from非常相似的逻辑实现的。 (还有一个额外的方法调用等待 ,但通常会在噪声中丢失)

在效率方面,删除订阅方法中的显式睡眠和轮询似乎是此设计中的主要目标。 不是在一段固定的时间内睡觉,最好是获得一个指示接收呼叫何时成功的未来,并且只有在接收数据时才运行订阅任务。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM