当 loop.run_until_complete 工作时 asyncio.run 失败

san*_*ash 1 python python-asyncio python-3.7 motor-asyncio

此代码失败:

import asyncio
from motor import motor_asyncio


_client = motor_asyncio.AsyncIOMotorClient()
_db = _client.db

users = _db.users


async def main():
    await users.create_index(
        'login',
        unique=True
    )


if __name__ == '__main__':
    #loop = asyncio.get_event_loop()
    #loop.run_until_complete(main())
    asyncio.run(main())
Run Code Online (Sandbox Code Playgroud)

出现此错误:

Traceback (most recent call last):
  File "/home/sanyash/myrepos/TKP/db.py", line 21, in <module>
    asyncio.run(main())
  File "/usr/local/lib/python3.7/asyncio/runners.py", line 43, in run
    return loop.run_until_complete(main)
  File "/usr/local/lib/python3.7/asyncio/base_events.py", line 584, in run_until_complete
    return future.result()
  File "/home/sanyash/myrepos/TKP/db.py", line 14, in main
    unique=True
RuntimeError: Task <Task pending coro=<main() running at /home/sanyash/myrepos/TKP/db.py:14> cb=[_run_until_complete_cb() at /usr/local/lib/python3.7/asyncio/base_events.py:158]> got Future <Future pending cb=[run_on_executor.<locals>._call_check_cancel() at /usr/local/lib/python3.7/motor/frameworks/asyncio/__init__.py:80]> attached to a different loop
Run Code Online (Sandbox Code Playgroud)

当我取消注释两行loop并注释时,asyncio.run它运行良好。什么事?我认为asyncio.run这是这两行的快捷方式。

问题出在motor_asyncio实现中,因为当我更改main为 simple 时return 42,它也asyncio.run运行良好。

Mik*_*mov 8

什么事?我认为asyncio.run这是这两行的快捷方式。

不,它做得更多。特别是它创建并设置了一个新的事件循环。这就是你得到错误的原因:AsyncIOMotorClient()为默认事件循环创建了一些异步的东西,但创建的另一个循环asyncio.run试图使用它。

如果你想保留asyncio.run你应该把初始化的东西移到里面main()

# ...

_client = None
_db = None
users = None


async def main():
    global _client, _db, users
    _client = motor_asyncio.AsyncIOMotorClient()
    _db = _client.db
    users = _db.users

    # ...
Run Code Online (Sandbox Code Playgroud)

一般来说,在事件循环已经设置并运行时开始工作而不是在模块级别做某事是一个好主意。