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

Python 使用asyncio和aiohttp的多个非阻塞任务,python,async-await,aiohttp,Python,Async Await,Aiohttp,我试图用asyncio和aiohttp执行几个非阻塞任务,但我认为这样做效率不高。我认为最好使用等待而不是屈服。有人能帮忙吗 def_init__(self): self.event_loop = asyncio.get_event_loop() def run(self): tasks = [ asyncio.ensure_future(self.subscribe()), asyncio.ensure_future(self.getServe

我试图用asyncio和aiohttp执行几个非阻塞任务,但我认为这样做效率不高。我认为最好使用等待而不是屈服。有人能帮忙吗

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

我必须收听服务器和订阅以收听广播,并根据广播的消息发布到不同的服务器。

等待不会比从中获得收益更有效。它可能更像蟒蛇,但是

async def foo():
    await some_future

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

就效率而言,删除订阅方法中的显式睡眠和轮询似乎是本设计的主要目标。与其睡一段固定的时间,不如得到一个未来,指示接收呼叫何时会成功,并且只有在接收有数据时才运行subscribe的任务。

根据:

wait与yield from类似,暂停执行read_数据 协同路由,直到db.fetch waitiable完成并返回结果 数据

它使用实现的收益率,还需要额外的验证步骤 它的论点。Wait仅接受可等待的,可为以下内容之一:

因此,我认为您的代码中没有效率问题,因为它们使用相同的实现

但是,我确实想知道为什么您返回
服务器
,但从不使用它

我在您的代码中看到的主要设计错误是,您同时使用了:

self.event_loop.run_until_complete(asyncio.gather(*tasks))
try:
   self.event_loop.run_forever()
据我所知,您只需要
永远运行()

一些额外提示:

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

try:
    loop.run_until_complete(asyncio.gather(*tasks))
finally:  # close the loop no matter what or you leak FDs
    loop.close()
我也用它来代替内置的,根据基准测试,它更有效

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

你能添加self.event\u loop.create\u服务器的定义(或者仅仅是签名)吗?它也是一个协程吗?它不是一个协程。我将把它添加到代码中对不起,这是一个协同程序:协同程序AbstractEventLoop.create_服务器(protocol_factory,host=None,port=None,*,family=socket.AF_unsec,flags=socket.AI_PASSIVE,sock=None,backlog=100,ssl=None,reuse_address=None,reuse_port=None)实现是一样的,所以我认为你是不对的。根据sleep,我不确定你所说的“此设计中的主要目标”是什么意思,但重点是暂停协同路由,所以这是一件好事。我使用的是aiohttp,必须从协同路由发送到服务器。创建客户机的示例代码使用async。如何调用异步函数?能否链接到示例代码?也许我没有为getServer使用协同路由是错误的。这是我正在使用的客户端的链接好的,我明白了,你确实需要在事件循环中创建服务器。但是,您仍然可以删除跑步记录,直到完成,并永久保留跑步记录。
import uvloop
...
    loop = uvloop.new_event_loop()
    asyncio.set_event_loop(loop)