python-asyncio TypeError:对象字典不能用于' await'表达

时间:2018-04-13 17:45:17

标签: python python-asyncio

我正在使用第三方模块从API检索数据。我只是想异步等待模块返回有时需要几秒钟的数据并冻结我的应用程序。但是,当我尝试等待对该模块的调用时,我收到TypeError:

TypeError: object dict can't be used in 'await' expression

import thirdPartyAPIwrapper

async def getData():
    retrienveData = await thirdPartyAPIWrapper.data()
    return await retrieveData

def main():
    loop = asncio.get_event_loop()
    data = loop.run_until_complete(getData())
    loop.close
    return data

为什么我不等待类型(' dict')?有没有解决的办法? 如果asyncio的async / await不能用于没有返回协程的第三方模块,那么我的其他选择是什么?

3 个答案:

答案 0 :(得分:9)

只能等待异步(用async def定义)函数。整个想法是,这些函数是以特殊的方式编写的,它们可以在不阻塞事件循环的情况下运行它们(await

如果你想从普通(用def定义)函数获得需要相当长时间才能执行的结果,你有以下选择:

  • 将整个函数重写为异步
  • 在另一个线程中调用此函数并等待异步结果
  • 在另一个进程中调用此函数并等待异步结果

通常你想选择第二个选项。

以下是如何操作的示例:

import asyncio
import time
from concurrent.futures import ThreadPoolExecutor


_executor = ThreadPoolExecutor(1)


def sync_blocking():
    time.sleep(2)


async def hello_world():
    # run blocking function in another thread,
    # and wait for it's result:
    await loop.run_in_executor(_executor, sync_blocking)


loop = asyncio.get_event_loop()
loop.run_until_complete(hello_world())
loop.close()

请阅读this answer,了解asyncio的工作原理。我认为它会对你有所帮助。

答案 1 :(得分:0)

由于thirdPartyAPIWrapper.data()是普通的同步功能,因此您应该在另一个线程中调用它。

asgiref库中有一个帮助函数。
假设我们有一个带有参数的阻塞函数:

import asyncio
import time

from asgiref.sync import sync_to_async


def blocking_function(seconds: int) -> str:
    time.sleep(seconds)
    return f"Finished in {seconds} seconds"

async def main():
    seconds_to_sleep = 5
    function_message = await sync_to_async(blocking_function)(seconds_to_sleep)
    print(function_message)

loop = asyncio.get_event_loop()
loop.run_until_complete(main())
loop.close()

该库中还有一个async_to_sync帮助函数。

答案 2 :(得分:0)

我正在编写测试用例,并且需要模拟异步功能。因此,您可以编写一个像这样的简单辅助函数。

 With ws2
         copyFrom.Copy .Range("A" & lrow + 5)

现在您可以等待任何事情

async def resolve(val):
    return val