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

文章探讨了如何优化企业IM应用中群消息的已读未读功能存储,以减少内存和磁盘空间占用。提出使用bitmap结合成员退出状态的标记来实现,通过将每个成员的已读未读状态由64位优化为2位,并记录退出成员状态,大大降低了存储成本。同时讨论了成员退出和重新加入时的处理策略,以及在maxid过大时的解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

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

530ec94fa89be99f7c55adc62992e4b9.png

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

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

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

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

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

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

 
 
struct UserInfo 
{ 
 uint64_t userid;
 uint32_t mapid;
};

struct GroupMetaInfo 
{
 vector <UserInfo> members;
 string name;
 uint32_t maxid;
 // other info
};

这样群成员每加入一个群里,就有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存储成本远超过最初的方案,可以用最初的方案来实现,客户端提前埋好兼容逻辑就可以了。

作者:小袁学习笔记

来源:www.toutiao.com/i6686735232772604429/

新建邮件:发送邮件、保存邮件、上传附件、发信设置 预定会议:发送邮件、保存邮件、上传附件、发信设置、会议设置 日程邀约:发送邮件、保存邮件、上传附件、发信设置 发起审批:发送邮件、保存邮件、上传附件、发信设置 导入文档新建:选择文档、搜索文档 发邮件:导入表格文件、指定发信字段 管理收件箱:删除邮件、彻底删除邮件、标为已/未读、设置星标、设置标签、拒收邮件、 举报邮件、移动到已发送邮件/文件夹、设置免提醒(移入/移出)、创建规则、 对收件人发起群聊、转发邮件、转发到聊天、导出邮件、打印邮件、 回复邮件 管理星标邮件:删除邮件、彻底删除邮件、标为已/未读、取消星标、设置标签、拒收邮件、 举报邮件、移动到已发送邮件/文件夹、设置免提醒(移入/移出)、创建规则、 对收件人发起群聊、转发邮件、转发到聊天、导出邮件、打印邮件、 回复邮件 管理草稿箱:再次编辑邮件、删除邮件、彻底删除邮件、导出邮件 管理已发送邮件:撤回邮件、再次编辑邮件、删除邮件、彻底删除邮件、标为已/未读、 设置星标、设置标签、拒收邮件、举报邮件、移动到收件箱/文件夹、 设置免提醒(移入/移出)、创建规则、对收件人发起群聊、转发邮件、 转发到聊天、导出邮件、打印邮件、回复邮件 管理已删除邮件:彻底删除邮件、标为已/未读、设置星标、设置标签、 拒收邮件、举报邮件、移动到已发送邮件/收件箱/文件夹、 设置免提醒(移入/移出)、创建规则、对收件人发起群聊、转发邮件、 转发到聊天、导出邮件、打印邮件、回复邮件 管理垃圾邮件:彻底删除邮件、标为已/未读、设置星标、设置标签、 拒收邮件、举报邮件、移动到已发送邮件/收件箱/文件夹、 设置免提醒(移入/移出)、创建规则、对收件人发起群聊、转发邮件、 转发到聊天、导出邮件、打印邮件、回复邮件、取消垃圾邮件 管理我的文件夹:文件夹重命名、新建文件夹、添加到常用文件夹、清空文件夹、删除文件夹 管理标签:新建标签、编辑标签、删除标签 搜索:全局搜索、分类搜索(设置筛选条件) 管理邮件联系人:添加邮件联系人(新建邮件联系人、批量导入邮件联系人) 搜索邮件联系人 编辑邮件联系人 删除邮件联系人 设置免提醒(移入/移出) 设置星标(添加/取消) 设置邮件特别提醒 邮件设置 这是企业微信邮件的用例 画用例图
03-12
<think>好的,我现在需要帮用户解决在zsh终端中找不到allure命令的问题。根据用户提供的引用内容,可能涉及环境变量配置的问题。首先,用户已经安装了allure-pytest,但执行时还是提示找不到命令,这说明可能环境变量没有正确设置。 根据引用[2],用户之前遇到类似的问题是因为环境变量添加到了.bash_profile而没有添加到.zshrc,因为系统默认使用zsh。所以同样的道理,allure的路径可能需要添加到.zshrc文件中。首先,我需要确认allure的安装路径是否正确,并且是否已经添加到PATH环境变量中。 用户可能通过Homebrew安装allure,或者直接下载了二进制包。如果是Homebrew安装的话,通常会自动配置环境变量,但如果是手动安装,可能需要手动添加路径。假设用户是手动安装,那么需要找到allure的可执行文件路径,比如/usr/local/allure/bin,然后将这个路径添加到.zshrc中。 另外,引用[2]提到在PyCharm的终端中可能仍然有问题,这时候需要执行source ~/.zshrc来刷新配置。所以用户可能需要检查是否在安装后执行了这个命令,或者在PyCharm的设置中是否正确配置了终端环境。 可能的解决方案步骤包括:1. 确认allure的安装位置;2. 将路径添加到.zshrc的PATH变量中;3. 执行source命令使配置生效;4. 检查PyCharm的终端设置。此外,用户可能没有正确安装allure命令行工具,而只安装了allure-pytest插件,这需要安装单独的allure命令行工具。 需要提醒用户,allure-pytest只是生成报告的插件,而生成报告需要allure命令行工具的支持。因此,用户可能需要单独安装allure,并确保其可执行文件在PATH中。安装方法可能包括使用Homebrew、SDKMAN或者下载二进制包。例如,用Homebrew安装的话,执行brew install allure,然后自动配置PATH。如果是手动下载,解压后需要将bin目录添加到PATH。 总结步骤:检查是否安装了allure命令行工具,确认安装路径,添加到.zshrc的PATH,执行source,验证安装。如果用户已经安装但路径未添加,就需要手动添加。同时,注意PyCharm可能需要重启或重新加载配置才能生效。</think>### 分步解决方案 **问题分析**:在zsh终端中提示`command not found: allure`,通常是由于以下原因导致: 1. Allure命令行工具未正确安装 2. Allure安装路径未添加到zsh的环境变量 3. 未刷新当前shell环境配置 --- #### **第一步:确认Allure命令行工具安装** 1. Allure-pytest仅生成测试报告数据,**必须单独安装Allure命令行工具** **安装方法**(任选其一): - **Homebrew安装**(推荐): ```bash brew install allure ``` - **手动安装**(需Java 8+环境): 1. 下载二进制包:https://github.com/allure-framework/allure2/releases 2. 解压到指定目录(如`/usr/local/allure`) 3. 添加环境变量到`.zshrc`: ```bash echo 'export PATH="$PATH:/usr/local/allure/bin"' >> ~/.zshrc ``` --- #### **第二步:验证环境变量配置** 1. 检查`.zshrc`文件是否包含Allure路径: ```bash cat ~/.zshrc | grep allure ``` 2. 若未找到相关配置,手动添加路径: ```bash echo 'export PATH="$PATH:/path/to/allure/bin"' >> ~/.zshrc ``` --- #### **第三步:刷新环境配置** 1. 立即生效配置: ```bash source ~/.zshrc ``` 2. 验证安装: ```bash allure --version ``` **成功输出示例**:`2.21.0` --- #### **第四步:解决PyCharm终端问题** 1. 在PyCharm中检查终端类型是否为`zsh`: `Settings -> Tools -> Terminal -> Shell path` 应为`/bin/zsh` 2. 重启PyCharm或重新加载终端 --- #### **验证完整流程** 1. 运行测试并生成报告: ```bash pytest --alluredir=./allure-results ``` 2. 查看报告: ```bash allure serve ./allure-results ``` --- ### 关键点总结 | 步骤 | 操作要点 | |----------------|------------------------------------------------------------------------| | **安装验证** | 必须安装Allure命令行工具,而非仅`allure-pytest`插件[^1] | | **环境变量** | 确保路径添加到`.zshrc`而非`.bash_profile`[^2] | | **配置刷新** | 执行`source ~/.zshrc`或重启终端 |
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值