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

How to terminate an async function in discord.py from another command

我正在使用 discord.ext.commands 制作一个 discord 机器人。我做了一个命令,即使有人被禁止,也会不断地取消禁止。这是代码:

@client.command(name="unban")
async def unban(ctx):
  while True:
    bans = await ctx.guild.bans()
    if(len(bans)>0):
      for ban in bans:
        ctx.guild.unban(ban)
    await asyncio.sleep(5)

但这是一个 while 循环,所以我想通过另一个命令(比如,stop_unban)终止这个函数。所以我想知道如何通过另一个函数(与stop_unban命令相关联)终止unban函数。

一种简单的方法是使用两个函数都可以访问的全局 bool 变量来控制禁止状态。
例如:

ban_state = False
@client.command(name="unban")
async def unban(ctx):
  global ban_state
  ban_state = True
  while ban_state:
    bans = await ctx.guild.bans()
    if(len(bans)>0):
      for ban in bans:
        await ctx.guild.unban(ban.user)
    await asyncio.sleep(5)
@client.command(name="stop_unban")
async def stop_unban(ctx):
  global ban_state
  ban_state = False
  await ctx.send('Unban Mode stopped')

但是,如果您希望解禁模式持续很长时间并且不使用全局变量,另一个可能更好的解决方案是使用 background task 而不是 while True。 例如:

from discord.ext import tasks
@tasks.loop(seconds=5.0)
async def unbanning(ctx):
    bans = await ctx.guild.bans()
    if(len(bans)>0):
      for ban in bans:
        await ctx.guild.unban(ban.user) #btw you need to await this and you have provide the user attribute of the ban not just the ban itself
    
@client.command(name="unban")
async def unban(ctx):
  unbanning.start(ctx)
@client.command(name="stop_unban")
async def stop_unban(ctx):
  unbanning.cancel()

Here 是关于后台任务的更多信息:)