python - 使用 discord.py 调用 python asyncio loop.run_until_complete() 不起作用?

标签 python python-3.x python-asyncio discord discord.py

使用 discord.py 制作 Discord 机器人,这是我第一次使用 asyncio,也可能是我第一次在 Python 中遇到如此令人沮丧的事情。

这个问题的重点不是教我如何使用 asyncio,而是教我如何避免使用它,即使它不是正确的做事方式。

所以我需要从常规 def 函数运行 discord 客户端协程。经过几个小时的搜索,我发现了这个:asyncio.get_event_loop().run_until_complete(...)。我设置了一个小脚本来测试它:

import asyncio

async def test():
    print('Success')

asyncio.get_event_loop().run_until_complete(test())

而且效果很好。所以我继续尝试在不和谐的机器人中使用它:

import discord
import asyncio

client = discord.Client()

@client.event
async def on_ready():
    test()

def test():
    asyncio.get_event_loop().run_until_complete(run())

async def run():
    print('Success')

client.run('TOKEN_HERE')

我得到一个错误... Stacktrace/Output:

Success
Ignoring exception in on_ready
Traceback (most recent call last):
  File "C:\Program Files\Python36\lib\site-packages\discord\client.py", line 307, in _run_event
    yield from getattr(self, event)(*args, **kwargs)
  File "C:/Users/OverclockedSanic/PyCharm Projects/asyncio test/test.py", line 8, in on_ready
    test()
  File "C:/Users/OverclockedSanic/PyCharm Projects/asyncio test/test.py", line 11, in test
    asyncio.get_event_loop().run_until_complete(run())
  File "C:\Program Files\Python36\lib\asyncio\base_events.py", line 454, in run_until_complete
    self.run_forever()
  File "C:\Program Files\Python36\lib\asyncio\base_events.py", line 408, in run_forever
    raise RuntimeError('This event loop is already running')
RuntimeError: This event loop is already running

奇怪的是最后的“成功”部分......我尝试了一些其他测试,看看我是否可以从协程返回数据或执行更多的东西,但它不能。

我什至尝试用 client.loop 替换 asyncio.get_event_loop(),但这也没有用。

我找了两天了,还是没有解决。有什么想法吗?

编辑:按照评论中的建议将 get_event_loop() 替换为 new_event_loop() 引发了以下问题:

Ignoring exception in on_ready
Traceback (most recent call last):
  File "C:\Program Files\Python36\lib\site-packages\discord\client.py", line 307, in _run_event
    yield from getattr(self, event)(*args, **kwargs)
  File "C:/Users/USER/PyCharm Projects/asyncio test/test.py", line 8, in on_ready
    test()
  File "C:/Users/USER/PyCharm Projects/asyncio test/test.py", line 11, in test
    asyncio.new_event_loop().run_until_complete(run())
  File "C:\Program Files\Python36\lib\asyncio\base_events.py", line 454, in run_until_complete
    self.run_forever()
  File "C:\Program Files\Python36\lib\asyncio\base_events.py", line 411, in run_forever
    'Cannot run the event loop while another loop is running')
RuntimeError: Cannot run the event loop while another loop is running

最佳答案

您的问题似乎本质上是关于混契约(Contract)步和异步代码。有两种可能:

1) 如果您的非异步例程不需要阻塞,只是安排一些异步任务(例如 send_message)稍后运行,那么它们可以简单地调用 get_event_loop( ).create_task().如果您希望在异步操作完成时调用其他(非异步)例程,您甚至可以对返回的任务使用 add_done_callback。 (如果要运行的例程也是非异步的,则使用 get_event_loop().call_soon()。)

2) 如果您的非异步例程绝对必须阻塞(包括可能等待异步例程),并且无法安排阻塞操作以备后用,那么您不应从与主事件循环相同的线程运行它们。您可以使用 concurrent.futures.ThreadPoolExecutor 创建一个线程池,并使用 asyncio.run_in_executor() 来安排您的非异步例程,然后等待结果。如果它们依次需要调用异步例程,那么 run_until_complete() 应该可以工作,因为现在您不是在已经具有事件循环的线程中运行。 (但要注意线程安全问题。如果您需要等待在主事件循环中运行某些东西,您可能需要类似 run_coroutine_threadsafe 的东西。)

如果有帮助,asgiref 包包含可以为您简化此过程的例程。它们的设计用途略有不同(Web 服务器),但也可能适合您。当您想从异步例程调用非异步例程时,您可以使用 await asgiref.sync.sync_to_async(func)(args),这将在线程池中运行该例程,然后使用 asgiref.sync.async_to_sync(func)(args) 当您想从线程池中运行的非异步例程调用异步例程时。

关于python - 使用 discord.py 调用 python asyncio loop.run_until_complete() 不起作用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46865451/

相关文章:

python - async - sync - 一个 python 事件循环中的异步调用

python - DIScord 命令认为 bool 为 True

python - Python 中的元组赋值,这是 Python 中的错误吗?

python - 如何等待协程直到满足条件?

python - pyinstaller kivy错误无法获取任何图像提供程序,中止

python - 在 python 中计算列表中的字符串然后过滤和匹配

python-3.x - 使用create_task()或collect()效率更高吗?

python - Cython header 指令和 __future__

python - 使用 pandas 格式化不一致的日期数据

python - 通过引用 Cython 传递单个整数?