运行一个函数会阻止机器人 - discord.py

Posted

技术标签:

【中文标题】运行一个函数会阻止机器人 - discord.py【英文标题】:Running a function blocks the bot - discord.py 【发布时间】:2021-12-27 20:30:46 【问题描述】:

我试图使用在 discord 上键入的命令从四个文件(它们有 4GB 大)中搜索数据。 搜索功能的执行时间大约需要十分钟,遍历4个文件;当我尝试连续输入两次命令时,第二个命令在第一个命令执行后执行(大约 10 分钟后)。

def main(self):
    @self.client.command()
    async def fsearch(ctx, *args):
        self.client.loop.create_task(self.searchTask(ctx, *args)) 

self.searchTask是搜索功能。

有什么想法可以让函数在不阻塞机器人的情况下执行吗? (一段时间后,机器人在执行过程中下线。)

更新:

这是来自self.searchTask函数的代码sn-p:

async def searchTask(self, ctx, *args):
    found = False
    name = args[0].lower().
    surname = args[1].lower()
    await ctx.send("Searching data...")
    with open(f"CURRENT_DIR/out.txt", "w") as fwrite:
        flist = os.listdir(f"CURRENT_DIR/datab")
        for x in flist:
            with open(f"CURRENT_DIR/datab/x", "rb") as f:
                for line in f:
            ...


   fwrite.close()
   if found:
       await ctx.send("Done! Here's the result:")
       await ctx.send(file=discord.File(f"CURRENT_DIR/out.txt"))
   else:
       await ctx.send("Can't find data")

【问题讨论】:

【参考方案1】:

您不能在异步代码中使用阻塞函数。似乎searchTask 正在阻塞,因此您的事件循环正在停止。 Asyncio 事件循环提供了一个简单的API 用于在线程池或进程池中运行阻塞函数,以防止它们阻塞您的事件循环。

编辑:示例:在线程中运行文件操作

def search_files():
    found = False
    flist = os.listdir(f"CURRENT_DIR/datab")
    for x in flist:
        with open(f"CURRENT_DIR/datab/x", "rb") as f:
            for line in f:
                found = found or is_found(line) # check your condition in is_found
    return found

async def searchTask(self, ctx, *args):
    name = args[0].lower()
    surname = args[1].lower()
    await ctx.send("Searching data...")
    loop = asyncio.get_running_loop()
    with open(f"CURRENT_DIR/out.txt", "w") as fwrite:
        found = await loop.run_in_executor(None, search_files)
        # write something to fwrite
        # also do this in executor if it's a lot

   fwrite.close() # this is unnecessary if you as using 'with'
   if found:
       await ctx.send("Done! Here's the result:")
       await ctx.send(file=discord.File(f"CURRENT_DIR/out.txt"))
   else:
       await ctx.send("Can't find data")

【讨论】:

self.searchTask 不是同步函数,是异步函数。 你能贴出它的代码吗? 完成,代码更新。 @ClownDown 好吧,代码仍然不完整,所以这很难确定,但我仍然相信searchTask 正在阻塞。您的代码不会自动异步工作,因为您使用async def 声明了它。只有当您的代码将控制权返回给循环时,事件循环才能退出您的代码,例如await。如果两个awaits 之间有大量计算,那么您的事件循环会在这段时间内被阻塞。 你可以,如果你重构出阻塞位。我将尝试使用基于您的代码的示例来更新我的答案。

以上是关于运行一个函数会阻止机器人 - discord.py的主要内容,如果未能解决你的问题,请参考以下文章

Discord bot 运行命令两次 discord.py

如何修复 Discord.py 不为语音命令运行我的异步功能?

关闭 Discord 机器人连接而不终止命令行 (discord.py)

在python中同时运行多个协程[Discord.py]

如何从另一个命令终止 discord.py 中的异步函数

带有 bio 的 discord.py 自定义配置文件