管理本地会话和消息

大约 9 分钟

管理本地会话和消息

本文介绍环信即时通讯 IM SDK 如何管理本地会话和消息。SDK 内部使用 SQLite 保存本地消息,方便消息处理。

除了发送和接收消息外,环信即时通讯 IM SDK 还支持以会话为单位对本地的消息数据进行管理,如获取与管理未读消息、搜索和删除历史消息以及统计消息流量等。其中,会话是一个单聊、群聊或者聊天室所有消息的集合。用户需在会话中发送消息、查看或清空历史消息等操作。

技术原理

环信即时通讯 IM Android SDK 支持管理用户设备上存储的消息会话数据,其中包含如下主要方法:

  • EMChatManager.getAllConversationsBySort 获取本地所有会话;
  • EMConversation.getAllMessages 从数据库中读取指定会话的消息;
  • EMConversation.getUnreadMsgCount 获取指定会话的未读消息数;
  • EMChatManager.getUnreadMessageCount 获取所有会话的未读消息数;
  • EMChatManager.markAllConversationsAsRead 指定会话的未读消息数清零;
  • EMChatManager.deleteConversation 删除会话及历史消息;
  • EMConversation.getUnreadMsgCount 获取指定会话的未读消息数;
  • EMChatManager.getMessage 根据消息 ID 获取消息;
  • EMChatManager.searchMsgFromDB(Type type, long timeStamp, int maxCount, String from, EMConversation.EMSearchDirection direction) 获取指定会话中特定类型的消息;
  • EMChatManager.searchMsgFromDB(long startTimeStamp, long endTimeStamp, int maxCount) 获取指定时间段内发送或接收的消息;
  • searchMsgFromDB(string keywords, long timeStamp, int maxCount, string from, EMSearchDirection direction) 根据关键字搜索本地会话消息;
  • EMChatManager.importMessages 批量导入消息到数据库;
  • EMChatManager.insertMessage 在指定会话中插入消息;
  • EMConversation.updateMessage 更新消息到本地数据库;
  • EMStatisticsManager.getMessageStatistics 根据消息 ID 获取消息流量统计信息;
  • EMStatisticsManager.getMessageCount 获取一定时间段内发送和/或接收的指定或全部类型的消息条数;
  • EMStatisticsManager.getMessageSize 获取一定时间段内发送和/或接收的指定或全部类型的消息的总流量。

前提条件

开始前,请确保满足以下条件:

  • 完成 SDK 初始化并连接到服务器,详见 快速开始
  • 了解环信即时通讯 IM API 的使用限制,详见 使用限制

实现方法

获取本地所有会话

你可以调用 getAllConversationsBySort 方法一次性获取本地所有会话。

SDK 从内存中获取会话,若未从本地数据库中加载过,会先从数据库加载到内存中。获取会话后,SDK 按照会话活跃时间(最新一条消息的时间戳)的倒序返回会话,置顶会话在前,非置顶会话在后,会话列表为 List<EMConversation> 结构。

提示

若使用该功能,需将 SDK 升级至 4.0.3。

示例代码如下:

List<EMConversation> conversations = EMClient.getInstance().chatManager().getAllConversationsBySort();

你也可以调用 getAllConversations 方法返回 Map <String, EMConversation> 结构的会话。

从数据库中读取指定会话的消息

你可以从本地数据库中读取指定会话的消息,示例代码如下:

EMConversation conversation = EMClient.getInstance().chatManager().getConversation(username);
// 获取此会话的所有消息。
List<EMMessage> messages = conversation.getAllMessages();
// SDK 初始化时,为每个会话加载 1 条聊天记录。如需更多消息,请到数据库中获取。该方法获取 `startMsgId` 之前的 `pagesize` 条消息,SDK 会将这些消息自动存入此会话,app 无需添加到会话中。
List<EMMessage> messages = conversation.loadMoreMsgFromDB(startMsgId, pagesize);

获取指定会话的未读消息数

你可以调用接口获取特定会话的未读消息数,示例代码如下:

EMConversation conversation = EMClient.getInstance().chatManager().getConversation(username);
conversation.getUnreadMsgCount();

获取所有会话的未读消息数

你可以通过接口获取所有会话的未读消息数量,示例代码如下:

EMClient.getInstance().chatManager().getUnreadMessageCount();

指定会话的未读消息数清零

你可以调用接口对特定会话的未读消息数清零,示例代码如下:

EMConversation conversation = EMClient.getInstance().chatManager().getConversation(username);
// 指定会话的未读消息数清零。
conversation.markAllMessagesAsRead();
// 将一条消息置为已读。
conversation.markMessageAsRead(messageId);
// 将所有未读消息数清零。
EMClient.getInstance().chatManager().markAllConversationsAsRead();

删除会话及历史消息

你可以删除本地会话和历史消息,示例代码如下:

// 删除和特定用户的会话,如果需要保留历史消息,传 `false`。
EMClient.getInstance().chatManager().deleteConversation(username, true);
// 删除指定会话中指定的一条历史消息。
EMConversation conversation = EMClient.getInstance().chatManager().getConversation(username);
conversation.removeMessage(deleteMsg.msgId);

删除服务端的会话及其历史消息,详见 删除服务端会话及其历史消息

根据消息 ID 获取消息

你可以调用 getMessage 方法根据消息 ID 获取本地存储的指定消息。如果消息不存在会返回空值。

// msgId:要获取消息的消息 ID。
EMMessage msg = EMClient.getInstance().chatManager().getMessage(msgId);

获取指定会话中特定类型的消息

你可以调用 searchMsgFromDB(Type type, long timeStamp, int maxCount, String from, EMConversation.EMSearchDirection direction) 方法从本地存储中获取指定会话中特定类型的消息。每次最多可获取 400 条消息。若未获取到任何消息,SDK 返回空列表。

//conversationId:会话 ID
EMConversation conversation = EMClient.getInstance().chatManager().getConversation(conversationId);
// Type:消息类型;timeStamp:消息搜索的起始时间戳,单位为毫秒。该参数设置后,SDK 从指定的时间戳的消息开始,按照搜索方向对消息进行搜索。若设置为负数,SDK 从当前时间开始,按消息时间戳的逆序搜索。
// maxCount:每次获取的消息数量,取值范围为 [1,400];direction:消息搜索方向:(默认)`UP`:按消息时间戳的逆序搜索;`DOWN`:按消息时间戳的正序搜索。
List<EMMessage> emMessages = conversation.searchMsgFromDB(EMMessage.Type.TXT, System.currentTimeMillis(), maxCount, from, EMConversation.EMSearchDirection.UP);

获取指定时间段内会话的消息

你可以调用 searchMsgFromDB(long startTimeStamp, long endTimeStamp, int maxCount) 方法从本地存储中搜索一定时间段内指定会话中发送和接收的消息。每次最多可获取 400 条消息。

//conversationId:会话 ID
EMConversation conversation = EMClient.getInstance().chatManager().getConversation(conversationId);
// startTimeStamp:搜索的起始时间戳;endTimeStamp:搜索的结束时间戳;maxCount:每次获取的消息数量,取值范围为 [1,400]。
List<EMMessage> messageList = conversation.searchMsgFromDB(startTimeStamp,endTimeStamp, maxCount);

根据关键字搜索会话消息

你可以调用 searchMsgFromDB(string keywords, long timeStamp, int maxCount, string from, EMSearchDirection direction) 方法从本地数据库获取会话中的指定用户发送的包含特定关键字的消息,示例代码如下:

//conversationId:会话 ID
EMConversation conversation = EMClient.getInstance().chatManager().getConversation(conversationId);
// keywords:搜索关键字;timeStamp:搜索的起始时间戳;maxCount:每次获取的消息数量,取值范围为 [1,400]。
// direction:消息搜索方向:(默认)`UP`:按消息时间戳的逆序搜索;`DOWN`:按消息时间戳的正序搜索。
List<EMMessage> messages = conversation.searchMsgFromDB(keywords, timeStamp, maxCount, from, EMConversation.EMSearchDirection.UP);

批量导入消息到数据库

如果你需要使用批量导入方式在本地会话中插入消息,可以使用下面的接口,构造 EMMessage 对象,将消息导入本地数据库。

示例代码如下:

EMClient.getInstance().chatManager().importMessages(msgs);

插入消息

如果你需要在本地会话中加入一条消息,比如收到某些通知消息时,可以构造一条消息写入会话。

例如插入一条无需发送但有需要显示给用户看的内容,类似 “XXX 撤回一条消息”、“XXX 入群”、“对方正在输入” 等。

示例代码如下:

// 将消息插入到指定会话中。
EMConversation conversation = EMClient.getInstance().chatManager().getConversation(username);
conversation.insertMessage(message);
// 直接插入消息。
EMClient.getInstance().chatManager().saveMessage(message);

更新消息到 SDK 本地数据库

如果需要更新消息可以选用以下方法的任意一种:

// 简洁方式:直接通过 `EMChatManager` 更新 SDK 本地数据库消息
EMClient.getInstance().chatManager().updateMessage(message);

// 正在使用 `EMConversation` 类时:先获取会话,再更新 SDK 本地数据库会话中的消息
EMConversation conversation = EMClient.getInstance().chatManager().getConversation(conversationId);
conversation.updateMessage(message);

获取本地消息的流量统计信息

本地消息的流量统计功能默认关闭。若要使用该功能,需在 SDK 初始化前设置 EMOptions#setEnableStatistics(boolean) 开启。

SDK 只支持统计该功能开启后最近 30 天内发送和接收的消息。各类消息的流量计算方法如下:

  • 对于文本、透传、位置消息,消息流量为消息体的流量;
  • 对于图片和视频消息,消息流量为消息体、图片或视频文件以及缩略图的流量之和;
  • 对于文件和语音消息,消息流量为消息体和附件的流量。

提示

  1. 统计时间段的开始时间和结束时间均为服务器接收消息的 Unix 时间戳。
  2. 对于携带附件的消息,下载成功后 SDK 才统计附件的流量。若附件下载多次,则会对下载的流量进行累加。
  3. 对于从服务器拉取的漫游消息,如果本地数据库中已经存在,则不进行统计。

SDK 仅统计本地消息的流量,而非消息的实际流量。一般而言,该统计数据小于实际流量,原因如下:

  • 未考虑发送消息时通用协议数据的流量;
  • 对于接收到的消息,服务端会进行消息聚合,添加通用字段,而消息流量统计为各消息的流量,未考虑通用字段的流量消耗。

根据消息 ID 获取消息流量统计信息

你可以根据消息 ID 获取指定消息的统计信息。该方法返回的消息流量统计信息包括消息 ID、消息的发送方和接收方、消息体类型、会话类型、消息方向、消息流量大小(单位为字节)以及服务器收到该消息的 Unix 时间戳。

示例代码如下:

EMMessageStatistics statistics = EMClient.getInstance().statisticsManager().getMessageStatistics(messageId);

获取一定时间段内发送和/或接收的消息条数

你可以统计一定时间段内发送和/或接收的指定或全部类型的消息,示例代码如下:

int number = EMClient.getInstance().statisticsManager().getMessageCount(startTime, endTime, direct, style);

获取一定时间段内发送和/或接收的消息的总流量

你可以统计一定时间段内发送和/或接收的指定或全部类型的消息的总流量,流量单位为字节。

示例代码如下:

long size = EMClient.getInstance().statisticsManager().getMessageSize(startTime, endTime, direct, style);