Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >IM群聊消息的已读回执功能该怎么实现?

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

作者头像
JackJiang
发布于 2018-08-29 08:09:40
发布于 2018-08-29 08:09:40
5K0
举报
文章被收录于专栏:即时通讯技术即时通讯技术

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

1、前言

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

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

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

目前市面上主流的移动端IM里,提供了已读回执的主要有阿里的钉钉、网易的易信、阿里的旺旺,如下图所示:

旺旺

易信

钉钉

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

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

2、IM开发干货系列文章

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

IM消息送达保证机制实现(一):保证在线实时消息的可靠投递》 《IM消息送达保证机制实现(二):保证离线消息的可靠投递》 《如何保证IM实时消息的“时序性”与“一致性”?》 《IM单聊和群聊中的在线状态同步应该用“推”还是“拉”?》 《IM群聊消息如此复杂,如何保证不丢不重?》 《一种Android端IM智能心跳算法的设计与实现探讨(含样例代码)》 《移动端IM登录时拉取数据如何作到省流量?》 《通俗易懂:基于集群的移动端IM接入层负载均衡方案分享》 《浅谈移动端IM的多点登陆和消息漫游原理》 《IM开发基础知识补课(一):正确理解前置HTTP SSO单点登陆接口的原理》 《IM开发基础知识补课(二):如何设计大量图片文件的服务端存储架构?》 《IM开发基础知识补课(三):快速理解服务端数据库读写分离原理及实践建议》 《IM开发基础知识补课(四):正确理解HTTP短连接中的Cookie、Session和Token》 《IM群聊消息的已读回执功能该怎么实现?》(本文)

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

3、正文引言

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

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

IM消息送达保证机制实现(一):保证在线实时消息的可靠投递

IM消息送达保证机制实现(二):保证离线消息的可靠投递

如何保证IM实时消息的“时序性”与“一致性”?

IM群聊消息如此复杂,如何保证不丢不重?

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离线,期望未来拉取到离线消息。

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

1)A发出群消息;

2)server收到消息后,一来要将群消息落地,二来要查询群里有哪些群成员,以便实施推送;

3)对于群成员,查询在线状态;

4)对于在线的群成员,实施推送。

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

核心问题3:如何保证接收方一定收到群消息?

答:各个收到消息后,要修改各群成员的last_ack_msgid,以告诉系统,这一条消息确认收到了。

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

对于在线的群友,收到群消息后,第一时间会ack、修改last_ack_msgid。

对于离线的群友,会在下一次登录时,拉取未读的所有群离线消息,并将last_ack_msgid修改为最新的一条消息。

核心问题4:如果ack丢失,群友会不会拉取重复的群消息?

答:会,可以根据msgid在客户端本地做去重,即使系统层面收到了重复的消息,仍然可以保证良好的用户体验。

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

6、已读回执流程的设计

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

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

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

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

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

增加了已读回执逻辑后,群消息的流程会有细微的改变,见下图:

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

1)将群消息落地;

2)查询群里有哪些群成员,以便实施推送;

之外,还需要:

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

接收方修改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;

发送方轮询拉取已读回执。

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

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2018.05.23 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
群消息已读回执(这个diao),究竟是推还是拉?
微信用于个人社交,产品设计上,在线状态,强制已读回执都有可能暴露个人隐私,故微信并无相关功能。
架构师之路
2018/07/27
1.6K0
群消息已读回执(这个diao),究竟是推还是拉?
支持百万人超大群聊的Web端IM架构设计与实践
现在IM群聊产品多种多样,有国民级的微信、QQ,企业级的钉钉、飞书,还有许多公司内部的IM工具,这些都是以客户端为主要载体。而且群聊人数通常都是有限制,微信正常群人数上限是500,QQ2000人,收费能达到3000人,这里固然有产品考量,但技术成本、资源成本也是很大的因素。笔者的业务场景上正好需要一个迭代更新快、轻量级(不依赖客户端)、单群百万群成员的纯H5的IM产品。
JackJiang
2025/03/13
910
支持百万人超大群聊的Web端IM架构设计与实践
feed留,单聊群聊,系统通知,状态同步,到底是推还是拉?
可以理解为一个发布订阅业务,典型业务是微博(朋友圈)。你关注了姚晨的微博,姚晨发布了消息,你的主页能看到她最新发布的消息,这个场景是推送,还是拉取呢?
架构师之路
2018/07/27
1.4K0
群聊比单聊,凭什么复杂这么多?
群聊是多人社交的基本诉求,一个群友在群内发了一条消息,期望做到: (1)在线的群友能第一时间收到消息; (2)离线的群友能在登陆后收到消息; 群消息的实时性、可达性、离线消息的复杂度,要远高于单对单消息。 常见的群消息流程如何? 群业务的核心数据结构有两个。 群成员表: t_group_users(group_id, user_id) 画外音:用来描述一个群里有多少成员。 群离线消息表: t_offine_msgs(user_id, group_id, sender_id,time, msg_id, msg
架构师之路
2022/09/22
6690
群聊比单聊,凭什么复杂这么多?
移动端IM中大规模群消息的推送如何保证效率、实时性?
众所周之,群聊是移动端IM的服务端技术难点所在,难在哪?大量的群聊消息,是一条条推给群内成员还是可以使用什么样的优化策略?试想一个2000人大群,一条消息的发出,如果瞬间被扩散写成2000条一对一消息的投递,对于接收方而言不过是一条消息而已,而服务端是以对相对比单聊消息的2000倍处理压力后的结果。那么服务端在保证消息投递的同时,面对这么大的压力该如何解决好效率问题?解决不好效率问题那实时性就不能保证!
JackJiang
2018/08/23
1.5K0
移动端IM中大规模群消息的推送如何保证效率、实时性?
群消息这么复杂,怎么能做到不丢不重?
【需求缘起】 之前的文章更多的聊了单对单的消息投递: 《微信为什么不丢消息?》 《http如何像tcp一样实时的收消息?》 群聊是多人社交的基本诉求,不管是QQ群,还是微信群,一个群友在群内发了一条消息: (1)在线的群友能第一时间收到消息 (2)离线的群友能在登陆后收到消息 由于“消息风暴扩散系数”的存在(概念详见《QQ状态同步究竟是推还是拉?》),群消息的复杂度要远高于单对单消息。群消息的实时性,可达性,离线消息是今天将要讨论的核心话题。 【常见的群消息流程】 开始讲群消息投递流程之前,先介绍两个群业
架构师之路
2018/03/01
1.6K0
群消息这么复杂,怎么能做到不丢不重?
webim如何保证消息的可靠投递
《webim如何保证消息的可靠投递》 上一章和大家分享了webim消息的实时性问题 消息的可靠性,即消息的不丢失和不重复,也是im系统中的一个难点。当初qq在技术上(当时叫oicq)因为以下两点原因才打败了icq: 1)qq的消息投递可靠(消息不丢失,不重复) 2)qq的垃圾消息少(它antispam做得好,这也是一个难点,但不是本文重点讨论的内容) 今天,本文将用十分通俗的语言,来讲述webim系统中消息可靠性的问题。 一、报文类型 im的客户端与服务器通过发送报文(也就是网络包)来完成消息的传递,报文分
架构师之路
2018/03/01
1.5K1
webim如何保证消息的可靠投递
揭秘企业微信如何优化满足ToB新挑战?
作者:潘唐磊 腾讯WXG开发工程师 导语| 本文主要总结了企业微信消息系统的架构与设计,阐述了toB业务带来的一些难点,面临哪些挑战,以及设计方案的对比与分析。同时总结了后台开发的一些常用手段,实用于消息系统,解决相关问题。 名词解释 seq:自增长的序列号,每条消息对应一个 ImUnion:消息互通系统,用于企业微信与微信的消息打通 控制消息:不可见消息,复用消息通道的一种可靠通知机制 应用类消息:系统应用下发的消息 api消息:第三方应用下发的消息 appinfo:每条消息对应的唯一strid,全局唯
腾讯大讲堂
2021/07/06
1.4K0
零基础IM开发入门(四):什么是IM系统的消息时序一致性?
本文引用了沈剑《如何保证IM实时消息的“时序性”与“一致性”?》一文的图片和内容(由于太懒,图没重新画),原文链接在文末。
JackJiang
2020/11/04
1.3K0
零基础IM开发入门(四):什么是IM系统的消息时序一致性?
直播系统聊天技术(六):百万人在线的直播间实时聊天消息分发技术实践
本文由融云技术团队原创分享,原题“聊天室海量消息分发之消息丢弃策略”,内容有修订。
JackJiang
2022/01/06
2.4K0
直播系统聊天技术(六):百万人在线的直播间实时聊天消息分发技术实践
一个海量在线用户即时通讯系统(IM)的完整设计Plus
《一个海量在线用户即时通讯系统(IM)的完整设计》(以下称《完整设计》)这篇文章发出来之后有不少读者咨询问题,提出意见或建议。主要集中在模块拆分、协议、存储等方面。针对这些问题做个简单说明。
Criss@陈磊
2019/08/02
5.8K0
一个海量在线用户即时通讯系统(IM)的完整设计
移动端重点是移动端,支持IOS/Android系统,包括IM App,嵌入消息功能的瓜子App,未来还可能接入客服系统。
普通程序员
2019/10/23
3.2K0
一个海量在线用户即时通讯系统(IM)的完整设计
IM群聊消息究竟是存1份(即扩散读)还是存多份(即扩散写)?
上一篇文章《IM群聊消息的已读回执功能该怎么实现?》是说,“很容易想到,是存一份”,被网友们骂了,大家争论的很激烈(见下图)。
JackJiang
2018/08/29
1.7K0
群消息,究竟存1份还是多份?
上一篇文章《群消息已读回执,究竟是推还是拉?》说,“很容易想到,是存一份”,被网友们骂了。
架构师之路
2018/07/27
6770
群消息,究竟存1份还是多份?
一套亿级用户的IM架构技术干货(下篇):可靠性、有序性、弱网优化等
本文内容和编写思路是基于邓昀泽的“大规模并发IM服务架构设计”、“IM的弱网场景优化”两文的提纲进行的,感谢邓昀泽的无私分享。
JackJiang
2021/03/22
7150
5亿用户如何高效沟通?钉钉首次对外揭秘即时消息服务DTIM
作者 | 陈万红,张世梁,杨世泉,余秋宇,谈云兵 策划 | 褚杏娟 这是钉钉第一次对外揭秘 DTIM(DingTalk IM,钉钉即时消息服务)。我们从设计原理到技术架构、从最底层存储模型到跨地域的单元化,全方位地展现了 DTIM 在实际生产中遇到各种挑战与解决方案,期望为企业级 IM 的建设贡献一臂之力。 钉钉已经有 2100 万 + 组织、5 亿 + 注册用户在使用。DTIM 为钉钉用户提供即时消息服务,用于组织内外的沟通,这些组织包括公司、政府、学校等,规模从几人到百万人不等。DTIM 有着丰富
深度学习与Python
2023/03/29
1K0
5亿用户如何高效沟通?钉钉首次对外揭秘即时消息服务DTIM
基于实践:一套百万消息量小规模IM系统技术要点总结
本文由公众号“后台技术汇”分享,原题“基于实践,设计一个百万级别的高可用 & 高可靠的 IM 消息系统”,原文链接在文末。由于原文存在较多错误和不准确内容,有大量修订和改动。
JackJiang
2021/11/27
2.1K0
基于实践:一套百万消息量小规模IM系统技术要点总结
《基于实践,设计一个百万级别的高可用 & 高可靠的 IM 消息系统》
https://xie.infoq.cn/article/4061081a5ce66137a8c021994
后台技术汇
2022/05/28
2K0
《基于实践,设计一个百万级别的高可用 & 高可靠的 IM 消息系统》
IM消息送达保证机制实现(二):保证离线消息的可靠投递1、前言 2、学习交流3、IM消息送达保证系列文章4、消息接收方不在线时的典型消息发送流程5、典型离线消息表的设计以及拉取离线消息的过程6、上述流
本文的上篇《IM消息送达保证机制实现(一):保证在线实时消息的可靠投递》中,我们讨论了在线实时消息的投递可以通过应用层的确认、发送方的超时重传、接收方的去重等手段来保证业务层面消息的不丢不重。
JackJiang
2018/08/23
8230
IM消息送达保证机制实现(二):保证离线消息的可靠投递1、前言
2、学习交流3、IM消息送达保证系列文章4、消息接收方不在线时的典型消息发送流程5、典型离线消息表的设计以及拉取离线消息的过程6、上述流
IM技术分享:万人群聊消息投递方案的思考和实践
本文由融云技术团队原创分享,原题“技术实践丨万人群聊的消息分发控速方案”,为使文章更好理解,内容有修订。
JackJiang
2021/08/30
2.4K0
推荐阅读
群消息已读回执(这个diao),究竟是推还是拉?
1.6K0
支持百万人超大群聊的Web端IM架构设计与实践
910
feed留,单聊群聊,系统通知,状态同步,到底是推还是拉?
1.4K0
群聊比单聊,凭什么复杂这么多?
6690
移动端IM中大规模群消息的推送如何保证效率、实时性?
1.5K0
群消息这么复杂,怎么能做到不丢不重?
1.6K0
webim如何保证消息的可靠投递
1.5K1
揭秘企业微信如何优化满足ToB新挑战?
1.4K0
零基础IM开发入门(四):什么是IM系统的消息时序一致性?
1.3K0
直播系统聊天技术(六):百万人在线的直播间实时聊天消息分发技术实践
2.4K0
一个海量在线用户即时通讯系统(IM)的完整设计Plus
5.8K0
一个海量在线用户即时通讯系统(IM)的完整设计
3.2K0
IM群聊消息究竟是存1份(即扩散读)还是存多份(即扩散写)?
1.7K0
群消息,究竟存1份还是多份?
6770
一套亿级用户的IM架构技术干货(下篇):可靠性、有序性、弱网优化等
7150
5亿用户如何高效沟通?钉钉首次对外揭秘即时消息服务DTIM
1K0
基于实践:一套百万消息量小规模IM系统技术要点总结
2.1K0
《基于实践,设计一个百万级别的高可用 & 高可靠的 IM 消息系统》
2K0
IM消息送达保证机制实现(二):保证离线消息的可靠投递1、前言 2、学习交流3、IM消息送达保证系列文章4、消息接收方不在线时的典型消息发送流程5、典型离线消息表的设计以及拉取离线消息的过程6、上述流
8230
IM技术分享:万人群聊消息投递方案的思考和实践
2.4K0
相关推荐
群消息已读回执(这个diao),究竟是推还是拉?
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文