点击上方“Java之间”,选择“置顶或者星标”

你关注的就是我关心的!

作者:小猿学习笔记

一朋友和我讨论他前段时间面试某大公司的一题目 :

企业IM比如企业微信、钉钉里面的群消息的有个已读未读的功能,发送者刚发出消息时,当前群里其他群成员都是未读状态,陆陆续续有人看了这个消息,这时候消息的详情变成x人已读,y人未读,如下图所示,有具体的已读未读列表(万恶的功能,看到同事or老板的消息不能假装没看到了),每条消息对应一个唯一的messageid(uint64_t),每个用户对应一个唯一的userid(uint64_t),应该如何保存这个消息对应的已读未读详情呢?

我第一时间给出一个很简单粗暴的方案:

对于每一个messageid,存当前readids + unreadids,当群成员A已读某一条消息时,把A userid从unreadids移除写到readids上就好了,客户端更新到messageid对应的详情列表,就可以展示m人已读,n人未读

显然这么简单粗暴的方案面试官是不会满意的,追问有没有更好的方案呢?

仔细分析,按照目前的设计,每一条消息,已读未读详情就要占用8B * 群成员数的内存,如果一个活跃的200人大群,每发一条消息,已读未读就要1600B,如果平均每天消息量是1k,那每个这样的群,每天就要1.6MB磁盘空间,对于客户端来说,特别是手机端,占用磁盘空间是用户不能接受的,又不能把工作消息删了,对于服务器端来说,用户群体如果特别大,那数据库存储这个成本也不小

其实未读已读就是一个0/1的标记而已,可以维护一个bitmap来实现呢?具体应该怎么做呢?

群元信息保存userid到自增mapid的映射

这样群成员每加入一个群里,就有mapid<->usreid的双向映射了,假如群里有5个成员ABCDE, 那就对应mapid 1-5,messageid对应的消息详情存储就可以设计成

{ uint32_t maxid, uint8_t readbit[]}

如上面的案例就是{5, readbit[0] =bin(0000 0000)}; 就占用了5B(4+1),A发消息,D已读消息时,就更新成{5,readbit[0]= bin(0000 1000)},其余4人都已读消息时 更新为{5, readbit[0]=bin(0001 1110)}

这是个粗略的方案,里面还有一些细节值得思考:

1、退出的成员呢?比如C退出群,发消息时maxid还是5,已读+未读总人数应该是3(不包括发消息者本人),目前信息只有5个bit(0/1),识别不出来谁已经退出群聊了

2、退出群聊的成员如何处理?从GruopMetaInfo里面删除么?退出群聊成员重新加入又如何分配id呢?

首先2这个点,退出群聊的成员只能标记删除,不能物理删除,不然客户端展示已读未读详情时,通过mapid找不到对应的userid,退出的成员又重新加入群聊这个就好办了,把标记删除改成非标记删除,还是用旧的mapid

至于1呢? 我目前想到比较好的方式就是再加多一个bitmap,记录成员在消息发送时是否已经退出群聊了,退出群聊就置为1, 所以最终方案就是

群信息增加userid,自增mapid双向映射,退出群聊成员标记删除,messageid 已读未读详情存储 {maxid, readbit[], quitbit[]}

新的方案带来怎样的收益呢?

1、增加自增mapid字段,一个群聊维护一份,成本几乎可以忽略不计

2、每个成员已读未读由8B(64bit)优化成2bit,减少62/64, 200人已读未读旧的方案1600B, 现在只需要(200/8) * 2 + 4 = 54 , 每条消息节约95%+

如果maxid如果到百万甚至千万级别,那岂不是灾难?一般实际场景,群聊是会限制人数的,就算不断踢人加新人,那maxid最多也只能到企业人数。 如果maxid达到一个特别大数字,已读未读对应的存储可以增加多一个flag,如果bitmap存储成本远超过最初的方案,可以用最初的方案来实现,客户端提前埋好兼容逻辑就可以了

原文链接:

https://m.toutiaocdn.cn/i6686735232772604429

最近热文阅读:

1、20年开发经验总结,“设计模型”才是架构设计的魂!

2、为什么架构师一定要懂“故障隔离”?

3、我是一条DQL

4、你真的理解零拷贝吗?

5、90%程序员面试都用得上的索引优化手册

6、90%架构师都知道的压力测试,你知道吗?

7、为什么程序员都不喜欢使用switch而使用if来做条件跳转

8、分享一些好用的 Chrome 扩展

关注公众号,你想要的Java都在这里!

面试题:群聊消息的已读未读设计相关推荐

  1. IM群聊消息的已读未读功能在存储空间方面的实现思路探讨

    1.引言 IM系统中,特别是在企业应用场景下,消息的已读未读状态是一个强需求. 以阿里的钉钉为例,钉钉的产品定位是用于商务交流,其"强制已读回执"功能,让职场人无法再"假 ...

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

    欢迎关注方志朋的博客,回复"666"获面试宝典 一朋友和我讨论他前段时间面试某大公司的一题目 : 企业IM比如企业微信.钉钉里面的群消息的有个已读未读的功能,发送者刚发出消息时,当 ...

  3. IM消息重试机制Java实现_IM群聊消息的已读回执功能该怎么实现?

    本文引用了架构师之路公众号作者沈剑的文章,内容有改动,感谢原作者. 1.前言我们平时在使用即时通讯应用时候,每当发出一条聊天消息,都希望对方尽快看到,并尽快回复,但对方到底有没有真的看到?我却并不知道 ...

  4. 大公司面试考细节,设计群聊消息的已读未读功能你说说怎么做?

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

  5. im即时通讯开发:IM群聊消息的已读回执功能

    我们平时在使用即时通讯应用时候,每当发出一条聊天消息,都希望对方尽快看到,并尽快回复,但对方到底有没有真的看到?我却并不知道. 一个残酷的现实是,很多时候对方其实是早就已经看到了这条消息,但出出种种原 ...

  6. IM即时通讯开发群聊消息的已读回执功能该怎么实现?

    我们平时在使用即时通讯应用时候,每当发出一条聊天消息,都希望对方尽快看到,并尽快回复,但对方到底有没有真的看到?我却并不知道.一个残酷的现实是,很多时候对方其实是早就已经看到了这条消息,但出出种种原因 ...

  7. 群聊消息“已读”/“未读” 功能解决方案!

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

  8. vue+websocket+nodejs实现聊天室 - 消息已读未读

    前言 上一篇讲了聊天室增加一对一单聊,这次讲如何新增已读未读状态. 大概思路: 服务器返回消息列表,增加参数status为1,单聊和群聊通过过滤状态区分对应未读数量,当前聊天,直接在currentMe ...

  9. netty之微信-群聊消息的收发及 Netty 性能优化(二十)

    群聊消息的收发及 Netty 性能优化 通过前面小节的学习,相信大家看到本小节标题就已经知道该如何实现本小节的功能了吧,为了给大家学到更多的知识,在实现了群聊消息收发之后,本小节将带给大家更多的惊喜. ...

最新文章

  1. 基于TensorRT优化的Machine Translation
  2. Bootstrap表单验证插件bootstrapValidator使用方法整理
  3. 使用varnish + nginx + lua搭建网站的降级系统
  4. bzoj3223 splay
  5. CDH5之Unexpected error.Unable to verify database connection
  6. Maven快速导出maven工程的依赖包
  7. TCP局域网 通讯 的消息发送
  8. linux属性表示的文件,Linux基础入门:文件和目录属性的含义
  9. REDIS 在电商中的实际应用场景(转)
  10. TensorFlow 第四步 多层神经网络 Mnist手写数字识别
  11. 【报告分享】第七次全国人口普查主要数据情况报告.pdf(附下载链接)
  12. 深度卷积生成对抗网络(DCGAN)原理与实现(采用Tensorflow2.x)
  13. 【opencv450】Vibe算法进行前景检测
  14. 微机原理是微型计算机与接口技术吗,《微机原理与接口技术》课程教学大纲
  15. pandas读取excel遇见空值的处理
  16. w10电脑文件不能放到服务器,w10设置文件服务器
  17. java 文本文件加密 文本文件解密
  18. loj #6570. 毛毛虫计数
  19. 锦鲤信小呆翻车!一手好牌打得稀烂背后:靠运气赚来的,终究会凭实力亏掉...
  20. Linux设置密码dictionary,Linux中修改密码出现it is based on a dictionary word解决方法

热门文章

  1. a到z的ascii码值是多少_c语言 ASCLL码中 A~Z和a~z是多少
  2. 树的序列化——浅谈 dfn 与欧拉序列
  3. 计算机专业兴趣小组 名字,计算机人兴趣小组口号
  4. Java jdt 编辑_使用JDT转java代码为AST
  5. 使html轮廓颜色不同,css中border颜色不同怎么设置?
  6. 黑白图片复原为彩色Picture Colorizer(图片着色器)
  7. edvac是商用计算机吗,EDVAC(eniac与edvac的区别)
  8. python常用的软件有哪些_Python编程常用软件有哪些?
  9. Ubuntu安装ESPNET(搞了一个多月)
  10. 使用有道云笔记的三个技巧