如何改进我的队列系统 - Discord.py

Posted

技术标签:

【中文标题】如何改进我的队列系统 - Discord.py【英文标题】:How can I improve my queue system - Discord.py 【发布时间】:2021-07-20 21:08:37 【问题描述】:

我创建了这个音乐机器人,我想改进我的队列命令,因为它目前不是很实用。每次我排队一首歌曲我都必须使用播放命令来播放它,但我想自动播放下一首歌曲,在播放命令中实现队列命令也很酷,但我不知道如何去做吧。 可以帮忙吗?

youtube_dl.utils.bug_reports_message = lambda: ''

ytdl_format_options = 
    'format': 'bestaudio/best',
    'outtmpl': '%(extractor)s-%(id)s-%(title)s.%(ext)s',
    'restrictfilenames': True,
    'noplaylist': True,
    'nocheckcertificate': True,
    'ignoreerrors': False,
    'logtostderr': False,
    'quiet': True,
    'no_warnings': True,
    'default_search': 'auto',
    'source_address': '0.0.0.0' # bind to ipv4 since ipv6 addresses cause issues sometimes


ffmpeg_options = 
    'options': '-vn'


ytdl = youtube_dl.YoutubeDL(ytdl_format_options)

class YTDLSource(discord.PCMVolumeTransformer):
    def __init__(self, source, *, data, volume=0.5):
        super().__init__(source, volume)

        self.data = data

        self.title = data.get('title')
        self.url = data.get('url')

    @classmethod
    async def from_url(cls, url, *, loop=None, stream=False):
        loop = loop or asyncio.get_event_loop()
        data = await loop.run_in_executor(None, lambda: ytdl.extract_info(url, download=not stream))

        if 'entries' in data:
            # take first item from a playlist
            data = data['entries'][0]

        filename = data['url'] if stream else ytdl.prepare_filename(data)
        return cls(discord.FFmpegPCMAudio(filename, **ffmpeg_options), data=data)

queue = []


@client.command(name='queue', help='This command adds a song to the queue')
async def queue_(ctx, url):
    global queue

    queue.append(url)
    await ctx.send(f'`url` added to queue!')



@client.command(name='play', help='This command plays songs')
async def play(ctx):
    global queue

    server = ctx.message.guild
    voice_channel = server.voice_client

    async with ctx.typing():
        player = await YTDLSource.from_url(queue[0], loop=client.loop, stream=True)
        voice_channel.play(player, after=lambda e: print('Player error: %s' % e) if e else None)

    await ctx.send('**Now playing:** '.format(player.title))
    del(queue[0])

编辑:所以我尝试做这样的事情但它不起作用,当我尝试使用 !play 当一首歌曲正在播放时它不会将它放入队列中,它说:ClientException: Already playing audio. 当这首歌完成了它说:TypeError: next() missing 2 required positional arguments: 'queue' and 'song' 这是代码:

queue = []

def next(client, queue, song):
    if len(queue)>0:
        new_song= queue[0]
        del queue[0]
        play(client, queue, new_song)

@bot.command()
async def join (ctx):
    member = ctx.author
    if not ctx.message.author.voice:
        await ctx.send(f"member.mention You are not connected to a voice channel  ❌")
    else:
        channel=ctx.message.author.voice.channel
        await channel.connect()


@bot.command(help="This command plays a song.")
async def play (ctx,*args):
    server = ctx.message.guild
    voice_channel= server.voice_client
    url=""
    for word in args:
        url+=word
        url+=''

    async with ctx.typing():
        player = await YTDLSource.from_url(url ,loop=bot.loop, stream=True)
        queue.append(player)
        voice_channel.play (player, after=lambda e: next(ctx))
        
    await ctx.send(f"**Now playing:** player.title")

【问题讨论】:

this 回答你的问题了吗? 您可以尝试 OOP 方法并编写一个 Queue 类,这将提供更多功能/帮助。 我认为它可以工作,但我不知道如何在我的代码中实现它,因为我实际上是流式传输歌曲而不下载它,而且因为我是编程新手,所以我不知道该怎么做...@Mr_Spaar 不知道怎么办,能帮忙吗??? @Xiddoc。我一直在尝试@Mr_Spaar 方法,但我不知道如何实现它。 【参考方案1】:

在您的异步 play 函数中,您有以下代码行:

voice_channel.play(player, after=lambda e: print('Player error: %s' % e) if e else None)

如您所见,有一个参数可用于在机器人完成向语音通道播放音频后执行操作。我建议您使用after= 参数更改代码以递归播放下一首歌曲,而不是按照您目前的方式使用它。您可以通过将 lambda 中的当前打印语句更改为再次异步调用 play 函数来做到这一点。

我还想指出,虽然学习如何使用 del() 函数非常适合学习如何优化代码,但这并不是您想要实现它的方式。在这种情况下,您应该改为.pop() 将值从队列中取出(就像这样:queue.pop(0)。您还应该知道弹出值返回值,因此您可以直接将其实现到的第一个参数中YTDLSource.from_url() 函数)。

我能提供的最后一点信息是你为你的队列系统编写了一个类。显然,这实际上并不会改变您解决问题所需要做的事情,但是学习如何编写 OOP 对象和类将对您将来有所帮助,并且由于它变得更有条理,它也使编写代码更容易。

总结一下:

首先删除del(queue[0]),然后将from_url() 方法中的queue[0] 参数值替换为queue.pop(0)。 最好使用 Queue 类更改整个 queue 数组,因为您可以很容易地向其中添加方法和其他有用的包装函数,但这不是必须的。 最后,代替lambda中的print语句(在.play()方法的after参数中),再次异步调用play函数。 (例如,像这样-lambda e: await play(ctx)

【讨论】:

当我将 voice_channel.play(player, after=lambda e: print('Player error: %s' % e) if e else None) 更改为 voice_channel.play(player, after=lambda e: await play(ctx)) 它给我一个语法错误:await outside async function 查看this。如果我的回答对您有帮助,请务必点赞并将其标记为有效(答案左侧的复选标记)! 我编辑了我的代码但我有一个错误,请你看一下好吗???

以上是关于如何改进我的队列系统 - Discord.py的主要内容,如果未能解决你的问题,请参考以下文章

基于共享内存的无锁消息队列设计

改进和优化美团Flume日志收集

Python 再次改进版通过队列实现一个生产者消费者模型

如何创建队列系统以在zend框架2.2中发送电子邮件。

优先队列(存储结构数组)--Java实现

Bellman-Ford算法的改进---SPFA算法