0

0

Discord.py 机器人获取用户头像命令教程

花韻仙語

花韻仙語

发布时间:2025-09-03 13:37:01

|

349人浏览过

|

来源于php中文网

原创

Discord.py 机器人获取用户头像命令教程

本教程详细指导如何在 discord.py 机器人中实现一个命令,以获取被提及用户的个人资料图片(头像)。文章首先展示在 on_message 事件中实现该功能的基本方法,随后重点介绍如何使用 discord.ext.commands 模块构建更结构化、易于维护的机器人,并提供完整的示例代码和重要注意事项,帮助开发者创建功能强大且清晰的 Discord 机器人。

实现用户头像获取命令 (!pfp)

在 discord.py 中,为机器人添加新功能通常涉及到处理用户消息。最直接的方法是在 on_message 事件监听器中检查消息内容。对于获取用户头像的需求,我们需要:

  1. 检查消息是否以特定命令前缀(例如 !pfp)开头。
  2. 从消息中解析出被提及的用户。
  3. 获取该用户的头像 URL。
  4. 将头像 URL 发送回频道。

以下是在 on_message 函数中实现 !pfp 命令的基本代码片段:

import discord

# ... (其他导入和机器人初始化代码) ...

@client.event
async def on_message(message):
    # 确保机器人不响应自己的消息
    if message.author == client.user:
        return

    # 检查是否为 !pfp 命令
    if message.content.startswith('!pfp'):
        # 检查消息中是否有被提及的用户
        if message.mentions:
            user = message.mentions[0]  # 获取第一个被提及的用户
            pfp_url = user.avatar.url  # 获取用户的头像 URL
            await message.channel.send(pfp_url)
        else:
            await message.channel.send("请提及一个用户,例如:`!pfp @用户名`")
        return # 处理完命令后返回,避免执行后续的 on_message 逻辑

    # ... (此处可以放置其他非命令相关的 on_message 逻辑) ...
    # 例如:
    # username = str(message.author)
    # user_message = str(message.content)
    # channel = str(message.channel)
    # print(f'{username} said: {user_message} ({channel})')
    # await send_message(message, user_message, is_private=False)

代码解析:

Mergeek
Mergeek

Mergeek是一个产品爱好者社区,专注于发现并介绍全球范围内的优质产品和项目

下载
  • message.mentions: 这是一个列表,包含消息中所有被提及的用户对象。我们通常取第一个 ([0])。
  • user.avatar.url: discord.User 或 discord.Member 对象都有一个 avatar 属性,它是一个 discord.Asset 对象。通过访问其 url 属性,可以直接获取用户头像的 CDN 链接。如果用户没有自定义头像,它将返回默认头像的 URL。

虽然这种方法能够实现功能,但将所有命令逻辑都堆砌在 on_message 函数中会导致代码难以维护和扩展。当机器人功能增多时,on_message 会变得非常臃肿。

优化机器人结构:使用 discord.ext.commands

为了更清晰、模块化地管理机器人命令,discord.py 提供了 discord.ext.commands 扩展。它允许我们定义独立的命令函数,并通过装饰器 @bot.command() 注册它们。

核心优势:

  • 自动解析命令: commands 模块会自动解析消息内容,识别命令前缀和参数。
  • 结构清晰: 每个命令都有自己的函数,易于理解和维护。
  • 参数转换: 可以指定命令参数的类型(例如 discord.Member),commands 模块会自动将用户输入转换为相应的对象。
  • 错误处理: 提供了更灵活的错误处理机制。

将 discord.Client 转换为 discord.ext.commands.Bot

首先,我们需要将 discord.Client 替换为 discord.ext.commands.Bot。Bot 类是 Client 的子类,因此它继承了 Client 的所有功能,并添加了命令处理能力。

import discord
from discord.ext import commands

# 定义命令前缀,例如 '!'
COMMAND_PREFIX = '!'

# 创建 Bot 实例
# 需要指定 intents,特别是 message_content 和 members
intents = discord.Intents.default()
intents.message_content = True # 允许机器人读取消息内容
intents.members = True         # 允许机器人获取成员信息(用于提及用户)

bot = commands.Bot(command_prefix=COMMAND_PREFIX, intents=intents)

@bot.event
async def on_ready():
    print(f'{bot.user} 已经上线并运行!')

# 如果需要处理非命令消息,可以保留 on_message 事件,但要确保不与命令处理冲突
@bot.event
async def on_message(message):
    if message.author == bot.user:
        return

    # 在这里处理非命令消息,例如:
    if "@everyone" in message.content:
        await message.channel.send(f"{message.author.mention}, 请勿提及所有人!")
        return # 处理完后返回,避免将此消息传递给命令处理器

    # 重要:将消息传递给命令处理器
    await bot.process_commands(message)

bot.process_commands(message) 的重要性: 当你在 on_message 事件中添加了自定义逻辑后,如果希望 commands.Bot 也能处理该消息中的命令,就必须调用 await bot.process_commands(message)。否则,你的自定义 on_message 逻辑会“吞掉”命令,导致 @bot.command() 注册的命令无法被触发。

使用 @bot.command() 实现 !pfp 命令

现在,我们可以使用 @bot.command() 装饰器来定义 !pfp 命令:

@bot.command(name='pfp', help='获取指定用户的头像。用法: !pfp [@用户]')
async def get_profile_picture(ctx, user: discord.Member = None):
    """
    获取被提及用户的头像。如果没有提及用户,则获取命令发送者的头像。
    """
    if user is None:
        user = ctx.author # 如果没有提及用户,则默认为命令发送者

    pfp_url = user.avatar.url
    embed = discord.Embed(
        title=f"{user.display_name} 的头像",
        color=discord.Color.blue()
    )
    embed.set_image(url=pfp_url)
    embed.set_footer(text=f"请求者: {ctx.author.display_name}")

    await ctx.send(embed=embed)

代码解析:

  • @bot.command(name='pfp', ...): 注册一个名为 pfp 的命令。用户可以通过 !pfp 来调用它。
  • async def get_profile_picture(ctx, user: discord.Member = None)::
    • ctx: 这是 commands.Context 对象,包含了命令的上下文信息,如频道、发送者、消息等。
    • user: discord.Member = None: 这是一个类型提示的参数。commands 模块会自动尝试将用户输入(例如 @用户名)转换为 discord.Member 对象。= None 表示这是一个可选参数,如果用户没有提及任何人,user 将为 None。
  • if user is None: user = ctx.author: 这是一个用户友好的改进,如果用户没有提及特定用户,则默认返回命令发送者的头像。
  • embed: 使用 discord.Embed 可以使头像展示更美观,包含标题、颜色和直接显示图片。embed.set_image(url=pfp_url) 会在 Discord 消息中直接嵌入图片。

完整示例代码 (使用 discord.ext.commands.Bot)

下面是一个整合了 Bot 结构、!pfp 命令和 on_message 事件处理的完整示例:

import discord
from discord.ext import commands
import requests # 假设你还需要天气功能

# 机器人配置
TOKEN = 'YOUR_BOT_TOKEN' # 替换为你的机器人令牌
WEATHER_API_KEY = 'YOUR_WEATHER_API_KEY' # 替换为你的天气 API 密钥
COMMAND_PREFIX = '!'

# 配置 Intents (意图)
# 必须启用 message_content 和 members
intents = discord.Intents.default()
intents.message_content = True
intents.members = True # 对于获取用户头像和成员信息是必需的

# 创建 Bot 实例
bot = commands.Bot(command_prefix=COMMAND_PREFIX, intents=intents)

# 机器人上线事件
@bot.event
async def on_ready():
    print(f'{bot.user} 已经上线并运行!')
    print(f'Bot ID: {bot.user.id}')
    print(f'Bot Name: {bot.user.name}')

# on_message 事件:处理非命令消息
@bot.event
async def on_message(message):
    # 确保机器人不响应自己的消息
    if message.author == bot.user:
        return

    # 打印消息日志
    username = str(message.author)
    user_message = str(message.content)
    channel = str(message.channel)
    print(f'{username} 在 {channel} 说: {user_message}')

    # 示例:处理 @everyone 提及
    if '@everyone' in message.content:
        await message.channel.send(f"{message.author.mention}, 请勿提及所有人!")
        # return # 如果希望处理完后不继续执行其他逻辑,可以取消注释

    # 示例:处理特定关键词
    if '重要' in user_message: # 假设 "önemli" 对应中文 "重要"
        if message.mentions:
            mentioned_user = message.mentions[0]
            if mentioned_user != bot.user:
                await mentioned_user.send(f'{message.author.display_name} 提醒您: {message.content}')
        # return

    # 重要:将消息传递给命令处理器,以便 @bot.command() 注册的命令能够被触发
    await bot.process_commands(message)

# !pfp 命令:获取用户头像
@bot.command(name='pfp', help='获取指定用户的头像。用法: !pfp [@用户]')
async def get_profile_picture(ctx, user: discord.Member = None):
    """
    获取被提及用户的头像。如果没有提及用户,则获取命令发送者的头像。
    """
    if user is None:
        user = ctx.author # 如果没有提及用户,则默认为命令发送者

    pfp_url = user.avatar.url
    embed = discord.Embed(
        title=f"{user.display_name} 的头像",
        color=discord.Color.blue()
    )
    embed.set_image(url=pfp_url)
    embed.set_footer(text=f"请求者: {ctx.author.display_name}")

    await ctx.send(embed=embed)

# 示例:!hava 命令(天气查询)
def get_weather(city):
    base_url = 'http://api.openweathermap.org/data/2.5/weather'
    params = {
        'q': city,
        'appid': WEATHER_API_KEY,
        'units': 'metric'
    }
    try:
        response = requests.get(base_url, params=params)
        response.raise_for_status() # 检查 HTTP 错误
        return response.json()
    except requests.exceptions.RequestException as e:
        print(f"天气查询失败: {e}")
        return None

def create_weather_embed(city, data):
    embed = discord.Embed(title=f"{city} 的天气状况:", color=0x00ff00)
    embed.add_field(name="天气", value=data['weather'][0]['description'], inline=False)
    embed.add_field(name="温度", value=f"{data['main']['temp']}°C", inline=True)
    embed.add_field(name="体感温度", value=f"{data['main']['feels_like']}°C", inline=True)
    embed.add_field(name="云量", value=f"{data['clouds']['all']}%", inline=True)
    embed.add_field(name="湿度", value=f"{data['main']['humidity']}%", inline=True)
    embed.set_thumbnail(url=f"http://openweathermap.org/img/w/{data['weather'][0]['icon']}.png")
    return embed

@bot.command(name='hava', help='查询指定城市的天气。用法: !hava [城市名]')
async def weather(ctx, *, city: str): # 使用 *city 表示获取所有剩余的参数作为城市名
    """
    查询指定城市的天气信息。
    """
    weather_data = get_weather(city)
    if weather_data:
        embed = create_weather_embed(city, weather_data)
        await ctx.send(embed=embed)
    else:
        await ctx.send(f"无法找到城市 '{city}' 的天气信息,或 API 出现错误。")

# 运行机器人
bot.run(TOKEN)

注意事项

  1. Intents (意图):

    • 在 discord.py 2.0+ 版本中,Intents 是强制性的。为了让机器人能够读取消息内容和获取成员信息(包括被提及的用户),必须在机器人初始化时启用 discord.Intents.message_content 和 discord.Intents.members。
    • intents.message_content = True 允许机器人访问消息的 content 属性。
    • intents.members = True 允许机器人获取服务器成员的详细信息,这对于 discord.Member 类型转换和访问 user.avatar 是必要的。
    • 请确保在 Discord 开发者门户中为你的机器人启用相应的 Privileged Gateway Intents。
  2. 机器人权限:

    • 确保你的机器人在其所在的 Discord 服务器中拥有发送消息的权限。如果使用 embed,还需要“嵌入链接”权限。
    • 如果涉及到私聊用户(例如 mentioned_user.send(...)),机器人需要能够私聊用户。
  3. 头像类型:

    • user.avatar.url 会自动返回用户当前头像的 URL,无论是自定义头像还是 Discord 默认头像。
    • 如果用户设置了动画头像(GIF),url 也会指向 GIF 链接。
  4. 错误处理:

    • 在生产环境中,应该为命令添加更健壮的错误处理机制。例如,可以使用 try-except 块来捕获潜在的 API 错误或网络问题
    • discord.ext.commands 提供了 on_command_error 事件,可以集中处理所有命令的错误。
  5. discord.Member 与 discord.User:

    • discord.Member 对象代表一个在特定服务器中的用户,它包含服务器特有的信息(如昵称、角色、加入日期)。
    • discord.User 对象代表 Discord 上的一个全局用户,不包含服务器特有信息。
    • 在服务器内,当提及用户时,commands 模块通常能解析为 discord.Member。在私聊或无法获取成员信息的场景下,可能会解析为 discord.User。user.avatar.url 对于两者都适用。

总结

通过本教程,你已经学会了如何在 discord.py 机器人中实现一个获取用户头像的命令。我们强调了从基础的 on_message 实现到推荐的 discord.ext.commands.Bot 结构的转变,后者提供了更清晰、更易于维护的代码组织方式。正确配置 Intents 和理解 commands.Context 对象是成功构建功能强大且稳定的 Discord 机器人的关键。现在,你可以基于这些知识进一步扩展你的机器人功能,为用户提供更多互动体验。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

583

2023.11.27

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

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

220

2023.12.07

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

773

2023.08.22

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

773

2023.08.22

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

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

395

2023.07.18

堆和栈区别
堆和栈区别

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

575

2023.08.10

C++类型转换方式
C++类型转换方式

本专题整合了C++类型转换相关内容,想了解更多相关内容,请阅读专题下面的文章。

299

2025.07.15

windows安全中心怎么关闭 windows安全中心怎么执行操作
windows安全中心怎么关闭 windows安全中心怎么执行操作

关闭Windows安全中心(Windows Defender)可通过系统设置暂时关闭,或使用组策略/注册表永久关闭。最简单的方法是:进入设置 > 隐私和安全性 > Windows安全中心 > 病毒和威胁防护 > 管理设置,将实时保护等选项关闭。

0

2026.01.26

2026年春运抢票攻略大全 春运抢票攻略教你三招手【技巧】
2026年春运抢票攻略大全 春运抢票攻略教你三招手【技巧】

铁路12306提供起售时间查询、起售提醒、购票预填、候补购票及误购限时免费退票五项服务,并强调官方渠道唯一性与信息安全。

1

2026.01.26

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.5万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

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

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