im群聊消息的已读回执功能该怎么实现?

imstudy imstudy     2022-11-16     321

关键词:

本文引用了架构师之路公众号作者沈剑的文章,内容有改动,感谢原作者。

1、前言


我们平时在使用即时通讯应用时候,每当发出一条聊天消息,都希望对方尽快看到,并尽快回复,但对方到底有没有真的看到?我却并不知道。

一个残酷的现实是,很多时候对方其实是早就已经看到了这条消息,但出出种种原因(大家都懂的),通常都是默默返回——假装没看见。

像微信这样的熟人社交工具,在产品的设计理念上,为了保持使用者的隐私性,在线状态、已读回执等涉及隐私的功能,都没有提供。但很多时候,尤其商务、办公场合下,特别需要一种强反馈的工具,这对于打造高效的团队很有帮助(虽然员工很反感,但老板都喜欢这样的功能,哈哈)。

目前市面上主流的移动端IM里,提供了已读回执的主要有阿里的钉钉、网易的易信、阿里的旺旺,如下图所示:
<ignore_js_op>技术分享图片    <ignore_js_op>技术分享图片    <ignore_js_op>技术分享图片 
▲ 上图从左至右分别为:钉钉、易信、旺旺(千牛)

以阿里的钉钉为例,钉钉的产品定位是用于商务交流,其“强制已读回执”功能,让职场人无法再“假装不在线”、“假装没收到”。更有甚者,钉钉的群聊“强制已读回执”功能,甚至能够知道谁读了消息,谁没有读消息(老板的福音啊)

那么群聊消息的收发流程、消息的送达保证、已读回执机制,到底该怎么实现呢?这就是今天要讨论的话题。

 

学习交流:

 

- 即时通讯开发交流3群:185926912[推荐]

- 移动端IM开发入门文章:《新手入门一篇就够:从零开发移动端IM

 

(本文同步发布于:http://www.52im.net/thread-1611-1-1.html

2、IM开发干货系列文章


本文是系列文章中的第14篇,总目录如下:


另外,如果您是IM开发初学者,强烈建议首先阅读《新手入门一篇就够:从零开发移动端IM》。

3、正文引言


首先我们需要了解一下群消息的设计、投递流程以及可达性保证机制,因不是本文要讨论的重点,所以尽量言简意赅,更详细的资料请见下方的推荐文章列表。

如您对聊天消息的投递和送达机制等尚无概念,可先读本系列文章的以下几篇,有助于您详细掌握这方面的内容:

 

4、群消息怎么设计?


大家一起跟着楼主的节奏,一步一步来看群消息怎么设计。

核心问题1:群消息,只存一份?还是,每个成员存一份?
答:存一份,为每个成员设置一个群消息队列,会有大量数据冗余,并不合适。

核心问题2:如果群消息只存一份,怎么知道每个成员读了哪些消息?
答:可以利用群消息的偏序关系,记录每个成员的last_ack_msgid(last_ack_time),这条消息之前的消息已读,这条消息之后的消息未读。该方案意味着,对于群内的每一个用户,只需要记录一个值即可。

解答上述两个核心问题后,很容易得到群消息的核心数据结构。

群消息表:记录群消息

group_msgs(msgid, gid, sender_uid, time, content);


各字段的含义为:消息ID,群ID,发送方UID,发送时间,发送内容。

群成员表:记录群里的成员,以及每个成员收到的最后一条群消息

group_users(gid, uid, last_ack_msgid);


各字段的含义为:群ID,群成员UID,群成员最后收到的一条群消息ID。

5、了解一下群消息发送的流程


在核心数据结构设计完之后,一起来看看群消息发送的流程(本系列中的文章《IM群聊消息如此复杂,如何保证不丢不重?》详细讲解了这个过程,可以深入读一读)。

业务场景:

  • 1)一个群中有A, uid1, uid2, uid3四名成员;
  • 2)A, uid1, uid2在线,期望实时收到在线消息;
  • 3)uid3离线,期望未来拉取到离线消息。


<ignore_js_op>技术分享图片 

其整个消息发送的流程1-4如上图:

  • 1)A发出群消息;
  • 2)server收到消息后,一来要将群消息落地,二来要查询群里有哪些群成员,以便实施推送;
  • 3)对于群成员,查询在线状态;
  • 4)对于在线的群成员,实施推送。


这个流程里,只要第二步消息落地完成,就能保证群消息不会丢失

核心问题3:如何保证接收方一定收到群消息?
答:各个收到消息后,要修改各群成员的last_ack_msgid,以告诉系统,这一条消息确认收到了。

在线消息,离线消息的last_ack_msgid的修改,又各有不同。

<ignore_js_op>技术分享图片 
对于在线的群友,收到群消息后,第一时间会ack、修改last_ack_msgid

<ignore_js_op>技术分享图片 
对于离线的群友,会在下一次登录时,拉取未读的所有群离线消息,并将last_ack_msgid修改为最新的一条消息

核心问题4:如果ack丢失,群友会不会拉取重复的群消息?
答:,可以根据msgid在客户端本地做去重,即使系统层面收到了重复的消息,仍然可以保证良好的用户体验。

上述流程,只能确保接收方收到消息,发送方仍然不知道哪些人在线阅读了消息,哪些人离线未阅读消息,并没有实现已读回执,那已读回执会对系统设计产生什么样的影响呢?

6、已读回执流程的设计


前面的基础知识我们已经了解的差不多,本节来讨论本文的重点内容,即群聊已读回执流程到底该怎么设计。

对于发送方发送的任何一条群消息,都需要知道,这条消息有多少人已读多少人未读,就需要一个基础表来记录这个关系

消息回执表:用来记录消息的已读回执

msg_acks(sender_uid, msgid, recv_uid, gid,if_ack);


各字段的含义为:发送方UID,消息ID,回执方UID,群ID,回执标记。

增加了已读回执逻辑后,群消息的流程会有细微的改变,见下图:
<ignore_js_op>技术分享图片 

接着,server收到消息后,除了要:

  • 1)将群消息落地;
  • 2)查询群里有哪些群成员,以便实施推送;


之外,还需要:

  • 3)插入每条消息的初始回执状态。


<ignore_js_op>技术分享图片 

接收方修改last_ack_msgid的流程,会变为:

  • 1)发送ack请求;
  • 2)修改last_ack_msgid,并且,修改已读回执if_ack状态;
  • 3)查询发送方在线状态;
  • 4)向发送方实时推送已读回执(如果发送方在线);


如果发送方不在线,ta会在下次登录的时候:

  • 5)从关联表里拉取每条消息的已读回执。


这里的初步结论是:

  • 如果发送方在线:会实时被推送已读回执;
  • 如果发送方不在线:会在下次在线时拉取已读回执。

 

7、已读回执流程优化方案


再次详细的分析下,群消息已读回执的“消息风暴扩散系数”,假设每个群有200个用户,其中20%的用户在线,即40各用户在线。

那么,群用户每发送一条群消息,会有:

  • 40个消息,通知给群友;
  • 40个ack修改last_ack_msgid,发给服务端;
  • 40个已读回执,通知给发送方。


可见,其消息风暴扩散系数非常之大

同时:

  • 需要存储40条ack记录。


群数量,群友数量,群消息数量越来越多之后,存储也会成为问题

是否有优化方案呢?

群消息的推送,能否改为接收方轮询拉取?
答:不能,消息接收,实时性是核心指标。

对于last_ack_msgid的修改,真的需要每个群消息都进行ack么?
答:其实不需要,可以批量ack,累计收到N条群消息(例如10条),再向服务器发送一次last_ack_msgid的修改请求,同时修改这个请求之前所有请求的已读回执,这样就能将40个发送给服务端的ack请求量,降为原来的1/10。

会带来什么副作用?
答:last_ack_msgid的作用是,记录接收方最近新取的一条群消息,如果不实时更新,可能导致,异常退出时,有一些群消息没来得及更新last_ack_msgid,使得下次登陆时,会拉取到重复的群消息。但这不是问题,客户端可以根据msgid去重,用户体验不会受影响

发送方在线时,对于已读回执的发送,真的需要实时推送么?
答:其实不需要,发送方每发一条消息,会收到40个已读回执,采用轮询拉取(例如1分钟一次,一个小时也就60个请求),可以大大降低请求量。
画外音:或者直接放到应用层keepalive请求里,做到0额外请求增加。

会带来什么副作用?
答:已读回执更新不实时,最坏的情况下,1分钟才更新回执。当然,可以根据性能与产品体验来折衷配置这个轮询时间。

如何降低数据量?
答:回执数据不是核心数据

  • 已读的消息,可以进行物理删除,而不是标记删除;
  • 超过N长时间的回执,归档或者删除掉。

 

8、本文小结


对于群消息已读回执,一般来说:

  • 如果发送方在线,会实时被推送已读回执;
  • 如果发送方不在线,会在下次在线时拉取已读回执。


如果要对进行优化,可以:

  • 接收方累计收到N条群消息再批量ack;
  • 发送方轮询拉取已读回执。


物理删除已读回执数据,定时删除或归档非核心历史数据。

(本文同步发布于:http://www.52im.net/thread-1611-1-1.html





































































































































面试官:群聊消息的已读未读功能,你来设计一个?

一朋友和我讨论他前段时间面试某大公司的一题目:企业IM比如企业微信、钉钉里面的群消息的有个已读未读的功能,发送者刚发出消息时,当前群里其他群成员都是未读状态,陆陆续续有人看了这个消息,这... 查看详情

头条面试官:如何设计群聊消息的已读未读功能?懵了。。(代码片段)

来源:https://www.toutiao.com/i6686735232772604429一朋友和我讨论他前段时间面试某大公司的一题目:企业IM比如企业微信、钉钉里面的群消息的有个已读未读的功能,发送者刚发出消息时,当前群里其他群成员都是未读状... 查看详情

面试官:群聊消息的已读未读功能,你来设计一个?

点击上方“朱小厮的博客”,选择“设为星标”后台回复"书",获取后台回复“k8s”,可领取k8s资料一朋友和我讨论他前段时间面试某大公司的一题目:企业IM比如企业微信、钉钉里面的群消息的有个已读未... 查看详情

在融云imkit会话界面基础上添加消息已读未读

...用过融云的同学们可能知道.融云IMkit的会话界面,发送玩消息后,如果对方已读,发送端则会显示小对号的图片.但是更具需求要把小对号改为已读未读.接下来我们就一块实现这个功能.打开回执功能首先,要确定打开融云的消息回执... 查看详情

基于netty,从零开发一个im即时通讯

...的IM服务端实现以下功能:    1)一对一的文本消息、文件消息通信;2)每个消息有“已发送”/“已送达”/“已读”回执;3)存储离线消息;4)支持用户登录,好友关系等基本功能;5&#... 查看详情

im系统实战

...)资源标准化确保消息体的可扩展性接收消息并解析显示群聊的特殊需求基础功能上无区别1个消息多个参与群聊的终端及时接收到服务器流量计 查看详情

im即时通讯功能解析:为什么微信里没有消息“已读”功能

为何其它IM里会有这个功能?换句话说:聊天消息的“已读”和“未读”状态在什么情况下该做呢? 这是一个典型的功能分析,遇到这种分析,我们应该如何用产品思维入手呢?第一步:结构性思维很... 查看详情

im即时通讯开发:高可用易伸缩高并发的im群聊单聊架构方案设计

要实现一整套能用于大用户量、高并发场景下的IM群聊,技术难度远超IM系统中的其它功能,原因在于:IM群聊消息的实时写扩散特性带来了一系列技术难题。举个例子:如一个2000人群里,一条普通消息的发出问题,将瞬间写扩... 查看详情

如何通过 Outlook 自动化获取属于已读回执 (ReportItem) 的 MailItem

...ReportItem对象。是否可以获得ID或属于给定已读回执的原始消息的一些详细信息?我查看了p 查看详情

PHP 邮件中的投递报告和已读回执

】PHP邮件中的投递报告和已读回执【英文标题】:DeliveryreportsandreadreceiptsinPHPmail【发布时间】:2010-11-0720:58:08【问题描述】:你们中的任何人都知道要添加哪些邮件标题才能获得已读回执和送达报告吗?现在是当您使用普通的PHP... 查看详情

openfire+smack怎么实现类似于qq群的功能?

...代码,找下查看原帖>> 参考技术B基于openfire+smack的IM群聊我是通过Openfire插件来实现的。基本思路就是通过插件拦截所有自定义的Message标签,然后由服务器向目前群内用户推送这条Message,来完成群聊的实现。希望对你有用。... 查看详情

亿级流量即时通讯im系统设计详解(全)

...展1.背景所需的功能?添加好友聊天会话列表单聊、群聊多终端登录(从数据库中拉取)消息漫游(一个终端收到很多消息后,登录另外一个终端实现同步)消息已读以及已读未读列表需考虑的约束条件... 查看详情

群聊内实现私聊功能

首先我们想到的是,消息发过来,我怎么知道是公聊消息还是私聊消息呢。所以,这里需要对消息进行处理,比如说在消息前后都加上一些特殊的字符,我们称为协议字符。为此,我们可以定义一个接口,专门来定义协议字符。... 查看详情

im系统:消息推送及离线存储

一、服务端维护每条消息对用户的送达状态  二、利用标号标记最新已读消息 说明:1、消息离线存储与拉取部分,有赞客服系统现有实现采用第二种方式2、文章参考了沈剑的《群消息这么复杂,怎么能做到不丢不重?... 查看详情

为自己搭建一个分布式im(即时通讯)系统

...用的消息终端,一个命令即可启动并向其他人发起通讯(群聊、私聊);同时内置了一些常用命令方便使用。整体的流程也比较简单,流程图如下:所以当我们自己部署时需要以下步骤:接下来重点看看具体的实现,比如群聊、... 查看详情

如何在 Quickblox 中为已读消息集成标记消息功能

】如何在Quickblox中为已读消息集成标记消息功能【英文标题】:HowtointegratemarkmessagefeatureforreadmessagesinQuickblox【发布时间】:2019-07-1111:10:50【问题描述】:我正在使用Chatviewcontroller或Quickblox来显示聊天详细信息。如果消息已读,... 查看详情

qt-实现qq群聊消息框

效果图如上:资源来自网络自己改了一下主要是窗口类和左消息类和右消息类,没有考虑设计模式,和代码重用,功能先实现了左消息类:1classCLeftBubbleWidget:publicQWidget2{3public:4CLeftBubbleWidget(QWidget*parent=NULL);5~CLeftBubbleWidget();6voidset... 查看详情

开发经验通知气泡实现思路

...,有一个功能叫通知气泡,就是提示用户有多少消息未读,然后让用户点击之后,气泡消失。        看似简单的功能,在设计起来很有讲究,因为,这种消息通知,肯定有私聊、广播、发送部... 查看详情