0

0

Discord.py 中避免 Presence 更新触发速率限制的正确实践

聖光之護

聖光之護

发布时间:2026-01-14 18:34:15

|

177人浏览过

|

来源于php中文网

原创

Discord.py 中避免 Presence 更新触发速率限制的正确实践

本文详解如何在 discord.py 中安全轮换 bot 活动状态(presence),规避因 `change_presence` 调用过于频繁导致的 websocket 速率限制警告(429 rate limited),重点修正 `asyncio.sleep` 同步误用、补充错误重试机制,并提供健壮、可长期运行的轮播方案。

在 Discord.py 中,通过 bot.change_presence() 动态更新 Bot 的在线状态(如“正在观看…”)是一种常见需求。但许多开发者会遇到如下警告:

WARNING  discord.gateway WebSocket in shard ID None is ratelimited, waiting 57.3 seconds

该警告表明:你的 Bot 已被 Discord 网关限流——并非因为 5 分钟间隔太短,而是因为代码中误用了 asyncio.sleep() 的同步形式,导致循环未真正暂停,从而在极短时间内重复发起请求,瞬间触达速率上限(Discord 对 /users/@me/settings 类操作有严格限频,通常为 5 次/5 分钟)

? 根本问题定位

原始代码中这一行是致命错误:

asyncio.sleep(300)  # ❌ 错误!这是协程对象,未 await,不产生实际延迟

它仅创建了一个 asyncio.sleep 协程对象,却未 await 执行,因此 while True 循环几乎零延迟地反复调用 change_presence(),等效于“疯狂刷请求”,远超 Discord 的容忍阈值。

Open Voice OS
Open Voice OS

OpenVoiceOS是一个社区驱动的开源语音AI平台

下载

✅ 正确写法必须是:

await asyncio.sleep(300)  # ✅ 真正挂起协程,让出控制权

✅ 推荐实现:带错误恢复的稳健轮播

以下是生产环境推荐的完整方案,已整合异常捕获与自动退避:

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}')

    # 使用后台任务(推荐)或 while 循环均可,此处保持简洁
    while True:
        try:
            activity = discord.Activity(
                type=discord.ActivityType.watching,
                name=random.choice(actaray)
            )
            await bot.change_presence(activity=activity)
            await asyncio.sleep(300)  # ✅ 正确 await,精确 5 分钟间隔
        except discord.HTTPException as e:
            if e.status == 429:  # 遇到限流
                retry_after = float(e.response.headers.get("Retry-After", "5"))
                print(f"Rate limited! Retrying after {retry_after:.1f}s...")
                await asyncio.sleep(retry_after + 1)  # 加 1 秒缓冲,避免边界重试
            else:
                print(f"HTTP error during presence update: {e}")
                await asyncio.sleep(10)  # 其他 HTTP 错误,降频重试
        except Exception as e:
            print(f"Unexpected error in presence loop: {e}")
            await asyncio.sleep(60)

# 可选:注册一个独立任务(更优雅,便于管理)
@bot.event
async def on_connect():
    if not hasattr(bot, '_presence_task') or bot._presence_task.done():
        bot._presence_task = bot.loop.create_task(_presence_rotator())

async def _presence_rotator():
    while True:
        await _update_random_presence()
        await asyncio.sleep(300)

async def _update_random_presence():
    try:
        await bot.change_presence(
            activity=discord.Activity(
                type=discord.ActivityType.watching,
                name=random.choice(actaray)
            )
        )
    except discord.HTTPException as e:
        if e.status == 429:
            retry_after = float(e.response.headers.get("Retry-After", "5"))
            await asyncio.sleep(retry_after + 1)

⚠️ 关键注意事项

  • 不要滥用 change_presence:Discord 明确限制用户级设置类操作(含状态更新)为 5 次/5 分钟/每个用户(Bot 账户即用户)。即使你设为 300s,若前序请求因网络延迟、重试失败而堆积,仍可能触发限流。
  • 永远 await 异步函数:asyncio.sleep()、bot.change_presence() 均为协程,必须 await,否则逻辑失效。
  • on_error 事件不可靠:Discord.py 的 on_error 并非总能捕获所有 HTTPException(尤其在 on_ready 内部抛出时),因此强烈建议在业务逻辑内直接 try/except,如上例所示。
  • 考虑使用 Activity 缓存或去重:若 actaray 列表较短,连续两次选中相同文案虽无害,但影响体验。可加入简单去重逻辑:
    last_name = None
    # 在循环内:
    name = random.choice([a for a in actaray if a != last_name] or actaray)
    last_name = name

✅ 总结

解决 Discord.py Presence 限流问题的核心在于:
修正 await asyncio.sleep() 用法,确保真实延时;
在 change_presence 调用周围包裹 try/except,主动处理 429 并遵循 Retry-After 头;
避免在 on_ready 中裸写无限循环,优先采用任务化(create_task)方式提升可维护性与可观测性。

遵循以上实践,你的 Bot 将稳定、安静地轮播状态,再也不会被网关警告“轰炸”。

相关专题

更多
while的用法
while的用法

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

85

2023.09.25

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

388

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

571

2023.08.10

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

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

18

2025.12.22

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

公务员递补名单公布时间 公务员递补要求
公务员递补名单公布时间 公务员递补要求

公务员递补名单公布时间不固定,通常在面试前,由招录单位(如国家知识产权局、海关等)发布,依据是原入围考生放弃资格,会按笔试成绩从高到低递补,递补考生需按公告要求限时确认并提交材料,及时参加面试/体检等后续环节。要求核心是按招录单位公告及时响应、提交材料(确认书、资格复审材料)并准时参加面试。

40

2026.01.15

公务员调剂条件 2026调剂公告时间
公务员调剂条件 2026调剂公告时间

(一)符合拟调剂职位所要求的资格条件。 (二)公共科目笔试成绩同时达到拟调剂职位和原报考职位的合格分数线,且考试类别相同。 拟调剂职位设置了专业科目笔试条件的,专业科目笔试成绩还须同时达到合格分数线,且考试类别相同。 (三)未进入原报考职位面试人员名单。

54

2026.01.15

国考成绩查询入口 国考分数公布时间2026
国考成绩查询入口 国考分数公布时间2026

笔试成绩查询入口已开通,考生可登录国家公务员局中央机关及其直属机构2026年度考试录用公务员专题网站http://bm.scs.gov.cn/pp/gkweb/core/web/ui/business/examResult/written_result.html,查询笔试成绩和合格分数线,点击“笔试成绩查询”按钮,凭借身份证及准考证进行查询。

11

2026.01.15

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

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

65

2026.01.14

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
swoole入门物联网开发与实战
swoole入门物联网开发与实战

共15课时 | 1.2万人学习

swoole项目实战(第二季)
swoole项目实战(第二季)

共15课时 | 1.2万人学习

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

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