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 的容忍阈值。

靠岸学术
靠岸学术

一款集翻译,阅读,文献管理于一体的英文文献阅读器

下载

✅ 正确写法必须是:

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 将稳定、安静地轮播状态,再也不会被网关警告“轰炸”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
while的用法
while的用法

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

107

2023.09.25

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

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

446

2023.07.18

堆和栈区别
堆和栈区别

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

606

2023.08.10

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

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

30

2025.12.22

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

142

2026.01.19

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

25

2026.03.13

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

44

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

174

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

50

2026.03.10

热门下载

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

精品课程

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

共15课时 | 1.4万人学习

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

共15课时 | 1.3万人学习

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

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