
在使用jda(java discord api)开发discord机器人时,开发者常常会遇到尝试获取频道历史消息,但event.getchannel().gethistory().getretrievedhistory()方法总是返回一个空列表的问题。这并不是因为没有历史消息,而是因为gethistory()方法返回的messagehistory对象在默认情况下并未立即填充(或从discord api中拉取)消息。
getRetrievedHistory()方法的作用是返回当前MessageHistory对象中已经被检索到的消息。如果JDA客户端尚未执行任何显式的消息获取操作,这个列表自然会是空的。它更像是一个缓存或一个容器,需要通过特定的方法来“填充”它。
要正确获取Discord频道的历史消息,关键在于使用MessageHistory对象提供的异步或同步获取方法,并确保这些操作完成。最直接且常用的方法是使用getHistoryFromBeginning(int count)结合.complete()。
getHistoryFromBeginning(int count)方法会创建一个请求,用于从频道的最初开始获取指定数量的消息。而.complete()是一个阻塞调用,它会暂停当前线程的执行,直到JDA成功从Discord API获取到消息并填充MessageHistory对象。一旦complete()方法执行完毕,MessageHistory对象就包含了请求到的历史消息,此时再调用getRetrievedHistory()就能获取到这些消息列表。
核心步骤:
以下代码演示了如何在JDA的onMessageReceived事件中正确地获取并打印频道历史消息。
import net.dv8tion.jda.api.entities.Message;
import net.dv8tion.jda.api.entities.channel.concrete.TextChannel;
import net.dv8tion.jda.api.entities.channel.ChannelType;
import net.dv8tion.jda.api.events.message.MessageReceivedEvent;
import net.dv8tion.jda.api.hooks.ListenerAdapter;
import net.dv8tion.jda.api.entities.MessageHistory;
import java.util.List;
public class MessageHistoryRetriever extends ListenerAdapter {
@Override
public void onMessageReceived(MessageReceivedEvent event) {
// 确保消息来自公会中的文本频道,避免在私聊或非文本频道中尝试获取历史消息
if (!event.isFromGuild() || event.getChannelType() != ChannelType.TEXT) {
System.out.println("消息非来自公会文本频道,跳过历史消息获取。");
return;
}
// 将频道转换为TextChannel类型,方便操作
TextChannel textChannel = event.getChannel().asTextChannel();
// 错误的获取历史消息示例 (原始问题中的代码逻辑)
// List<Message> emptyMessages = textChannel.getHistory().getRetrievedHistory();
// System.out.println("错误示例获取到的消息数量 (总是0): " + emptyMessages.size());
System.out.println("尝试从频道 [" + textChannel.getName() + "] 获取历史消息...");
try {
// 正确的获取历史消息方法:
// 1. 从TextChannel获取MessageHistory对象,并指定从频道开始获取最多100条消息。
// .complete() 是一个阻塞调用,会等待Discord API响应并填充MessageHistory对象。
MessageHistory messageHistory = textChannel.getHistoryFromBeginning(100).complete();
// 2. 从已填充的MessageHistory对象中检索消息列表。
List<Message> messages = messageHistory.getRetrievedHistory();
System.out.println("成功获取到历史消息数量: " + messages.size());
// 遍历并打印获取到的消息内容
for (int i = 0; i < messages.size(); i++) {
Message msg = messages.get(i);
System.out.println("消息 " + (i + 1) + " (ID: " + msg.getId() + ", 作者: " + msg.getAuthor().getName() + "): " + msg.getContentRaw());
}
// 也可以使用Java 8的forEach方法
// messages.forEach(msg -> System.out.println("消息: " + msg.getContentRaw()));
} catch (Exception e) {
System.err.println("获取历史消息时发生错误: " + e.getMessage());
e.printStackTrace();
}
}
}JDA中获取Discord频道历史消息的关键在于理解MessageHistory对象的生命周期和其消息填充机制。简单地调用getHistory().getRetrievedHistory()是不足以获取到消息的,因为它只是访问一个未填充的容器。正确的做法是利用getHistoryFromBeginning(count).complete()等方法,显式地触发从Discord API获取消息的操作,并等待其完成。掌握这一机制,将能有效地在JDA机器人中实现历史消息的检索功能。
以上就是JDA Discord机器人:解决获取频道历史消息为空的问题的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号