php中文网 | cnphp.com

 找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 452|回复: 0

消息已读、未读是怎么设计的?

[复制链接]

2672

主题

2679

帖子

9503

积分

管理员

Rank: 9Rank: 9Rank: 9

UID
1
威望
0
积分
6709
贡献
0
注册时间
2021-4-14
最后登录
2024-5-19
在线时间
674 小时
QQ
发表于 2022-5-3 10:12:48 | 显示全部楼层 |阅读模式
比如企业微信、钉钉里面的群消息的有个已读未读的功能,发送者刚发出消息时,当前群里其他群成员都是未读状态,陆陆续续有人看了这个消息,这时候消息的详情变成x人已读,y人未读。

每条消息对应一个唯一的 messageid(uint64_t),每个用户对应一个唯一的 userid(uint64_t),应该如何保存这个消息对应的已读未读详情呢?

其实未读已读就是一个0/1的标记而已,可以维护一个Bitmap来实现呢。

保存 userid 到自增 mapid 的映射
[mw_shl_code=applescript,true]struct UserInfo
[/mw_shl_code]
这样群成员每加入一个群里,就有 mapid<->usreid 的双向映射了。

假如群里有5个成员 ABCDE, 那就对应 mapid 1-5,messageid 对应的消息详情存储就可以设计成
[mw_shl_code=applescript,true]{ uint32_t maxid, uint8_t readbit[]}
[/mw_shl_code]
如上面的案例就是 {5, readbit[0] = bin(0000 0000)}; 就占用了 5B(4+1) 。

A发消息,D已读消息时,就更新成 {5,readbit[0]= bin(0000 1000)}。

其余4人都已读消息时,更新为 {5, readbit[0]=bin(0001 1110)} 。

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

退出的成员呢?

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

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

首先2这个点:

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

至于1呢:

可以再加多一个 Bitmap ,记录成员在消息发送时是否已经退出群聊了,退出群聊就置为1。

所以最终方案就是:

群信息增加userid,自增mapid双向映射,退出群聊成员标记删除。

messageid 已读未读详情存储 {maxid, readbit[], quitbit[]}

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

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

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

如果maxid如果到百万甚至千万级别,那岂不是灾难?

一般实际场景,群聊是会限制人数的,就算不断踢人加新人,那maxid最多也只能到企业人数。

如果maxid达到一个特别大数字,已读未读对应的存储可以增加多一个flag,如果bitmap存储成本远超过最初的方案,可以用最初的方案来实现,客户端提前埋好兼容逻辑。





上一篇:Java-GUI 编程之 Swing
下一篇:修复Arch Linux和Manjaro Linux无法显示emoji的问题
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|php中文网 | cnphp.com ( 赣ICP备2021002321号-2 )

GMT+8, 2024-5-20 00:56 , Processed in 0.152898 second(s), 32 queries , Gzip On.

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2020, Tencent Cloud.

申明:本站所有资源皆搜集自网络,相关版权归版权持有人所有,如有侵权,请电邮(fiorkn@foxmail.com)告之,本站会尽快删除。

快速回复 返回顶部 返回列表