0

0

Discord.py 中正确实现 Bot 状态轮换以避免速率限制的完整指南

花韻仙語

花韻仙語

发布时间:2026-01-14 18:41:23

|

574人浏览过

|

来源于php中文网

原创

Discord.py 中正确实现 Bot 状态轮换以避免速率限制的完整指南

本文详解如何在 discord.py 中安全地轮换 bot 的在线状态(如活动状态),避免触发 websocket 速率限制警告,重点修正 `asyncio.sleep` 未 await 导致的高频请求问题,并提供错误重试机制与最佳实践。

在 Discord.py 中,频繁调用 bot.change_presence()(例如在无限循环中)极易触发网关速率限制(Gateway Rate Limit),表现为控制台持续输出类似 WARNING discord.gateway WebSocket in shard ID None is ratelimited, waiting X seconds 的警告——即使你设置了 5 分钟(300 秒)间隔,问题仍存在,根本原因在于:asyncio.sleep(300) 本身是协程对象,若未用 await 调用,它不会真正暂停执行,而是立即返回并进入下一轮循环,导致 change_presence 实际以毫秒级频率被反复调用,远超 Discord 的允许阈值(通常为每 15 秒最多 1 次)

以下是修复后的推荐实现,兼顾稳定性、可维护性与健壮性:

萝卜简历
萝卜简历

免费在线AI简历制作工具,帮助求职者轻松完成简历制作。

下载
import asyncio
import random
import discord
from discord.ext import commands

actaray = [
    "PcktWtchr's Videos",
    "Cams",
    "and Listening Always",
    "or Listening or Both"
]

@bot.event
async def on_ready():
    print(f'Logged in as {bot.user}')
    # 启动后台任务(推荐方式,避免阻塞事件循环)
    bot.loop.create_task(presence_rotation())

async def presence_rotation():
    while True:
        try:
            activity = discord.Activity(
                type=discord.ActivityType.watching,
                name=random.choice(actaray)
            )
            await bot.change_presence(activity=activity)
            # ✅ 正确使用 await:真正暂停协程 300 秒
            await asyncio.sleep(300)
        except discord.HTTPException as e:
            if e.status == 429:  # 429 Too Many Requests
                retry_after = int(e.response.headers.get("Retry-After", "5"))
                print(f"Rate limited! Retrying after {retry_after} seconds...")
                await asyncio.sleep(retry_after + 1)  # 加 1 秒缓冲,避免边界重试失败
            else:
                print(f"HTTP error during presence update: {e}")
                await asyncio.sleep(60)  # 其他 HTTP 错误降频重试
        except Exception as e:
            print(f"Unexpected error in presence rotation: {e}")
            await asyncio.sleep(60)

# 可选:全局错误处理器(补充兜底)
@bot.event
async def on_error(event, *args, **kwargs):
    if event == "on_ready" and args and isinstance(args[0], discord.HTTPException):
        if args[0].status == 429:
            retry_after = int(args[0].response.headers.get("Retry-After", "5"))
            print(f"on_ready rate limit hit. Waiting {retry_after}s before retry...")
            await asyncio.sleep(retry_after + 1)

关键改进说明:

  • await asyncio.sleep(300) 替代 asyncio.sleep(300):这是最核心的修复。协程必须 await 才会挂起当前任务,否则循环体瞬间执行完毕,造成“伪延迟”。
  • 封装为独立异步任务:使用 bot.loop.create_task() 启动 presence_rotation(),比直接在 on_ready 中写 while True 更清晰,也便于后续扩展(如动态启停)。
  • 精细化异常捕获:单独处理 discord.HTTPException,尤其识别 429 状态码,并读取响应头中的 Retry-After 字段进行精准退避,而非盲目等待固定时间。
  • 防御性编程:对非 429 错误(如网络中断、认证失效)设置合理 fallback 延迟(如 60 秒),防止异常导致高频重试。

注意事项:

  • Discord 官方未公开精确的 change_presence 速率限制规则,但实测建议 最低间隔 ≥ 15 秒,5 分钟(300 秒)完全安全——前提是真正生效。
  • 避免在 on_ready 内直接写阻塞式 while True 循环(尤其未加 await),这会干扰事件循环调度,甚至影响其他事件响应。
  • 若 Bot 运行于多分片(shard)环境,请确保状态更新逻辑对所有分片一致,或使用 bot.wait_until_ready() 做前置校验。

通过以上改造,你的 Bot 将稳定、合规地轮换状态,彻底告别烦人的速率限制警告。

相关专题

更多
504 gateway timeout怎么解决
504 gateway timeout怎么解决

504 gateway timeout的解决办法:1、检查服务器负载;2、优化查询和代码;3、增加超时限制;4、检查代理服务器;5、检查网络连接;6、使用负载均衡;7、监控和日志;8、故障排除;9、增加缓存;10、分析请求。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

566

2023.11.27

default gateway怎么配置
default gateway怎么配置

配置default gateway的步骤:1、了解网络环境;2、获取路由器IP地址;3、登录路由器管理界面;4、找到并配置WAN口设置;5、配置默认网关;6、保存设置并退出;7、检查网络连接是否正常。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

219

2023.12.07

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

85

2023.09.25

Golang WebSocket与实时通信开发
Golang WebSocket与实时通信开发

本专题系统讲解 Golang 在 WebSocket 开发中的应用,涵盖 WebSocket 协议、连接管理、消息推送、心跳机制、群聊功能与广播系统的实现。通过构建实际的聊天应用或实时数据推送系统,帮助开发者掌握 如何使用 Golang 构建高效、可靠的实时通信系统,提高并发处理与系统的可扩展性。

18

2025.12.22

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

34

2026.01.14

php与html混编教程大全
php与html混编教程大全

本专题整合了php和html混编相关教程,阅读专题下面的文章了解更多详细内容。

14

2026.01.13

PHP 高性能
PHP 高性能

本专题整合了PHP高性能相关教程大全,阅读专题下面的文章了解更多详细内容。

33

2026.01.13

MySQL数据库报错常见问题及解决方法大全
MySQL数据库报错常见问题及解决方法大全

本专题整合了MySQL数据库报错常见问题及解决方法,阅读专题下面的文章了解更多详细内容。

18

2026.01.13

PHP 文件上传
PHP 文件上传

本专题整合了PHP实现文件上传相关教程,阅读专题下面的文章了解更多详细内容。

12

2026.01.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Django 教程
Django 教程

共28课时 | 3.1万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.1万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号