我正在使用第三方模块从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不能用于没有返回协程的第三方模块,那么我的其他选择是什么?
答案 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