消息~组件(群聊类型)ConcurrentHashMap发送

news2025/5/14 6:25:34

为什么选择ConcurrentHashMap?
在开发聊天应用时,我们需要存储和管理大量的聊天消息数据,这些数据会被多个线程频繁访问和修改。比如,当多个用户同时发送消息时,服务端需要同时处理这些消息的存储和查询。如果用普通的HashMap,可能会出现线程安全问题,比如数据被覆盖或者读取到错误的数据。
而ConcurrentHashMap是一个专门为多线程环境设计的数据结构,它的主要优点如下:

  1. 线程安全
    ConcurrentHashMap内部通过锁分段机制(或者在Java 8及以上版本中使用CAS操作和synchronized锁)来保证线程安全。这意味着多个线程可以同时读写它,而不会出现数据错乱的问题。
  2. 高性能
    相比普通的HashMap,ConcurrentHashMap在多线程环境下性能更高。它允许多个线程同时读取和更新数据,而不会像Collections.synchronizedMap那样锁住整个表。
  3. 支持高并发
    聊天应用通常需要处理大量的并发请求,比如同时接收和发送消息。ConcurrentHashMap能够很好地支持这种高并发场景,确保数据的读写操作不会成为性能瓶颈。
  4. 易于使用
    它的使用方式和普通的HashMap非常相似,几乎不需要额外的学习成本。只需要把HashMap替换为ConcurrentHashMap,就可以在多线程环境中安全地使用。
    举个例子
    假设我们有一个聊天应用,用户A和用户B同时向用户C发送消息。如果没有线程安全机制,可能会出现以下问题:
    ● 用户A的消息覆盖了用户B的消息。
    ● 用户C看到的消息顺序混乱。
    而ConcurrentHashMap可以很好地解决这些问题。它会确保每个消息都被正确存储,并且在多个线程同时操作时不会出现冲突。

总结
选择ConcurrentHashMap是因为它既安全又高效,特别适合聊天应用这种需要处理大量并发数据的场景。它能帮我们省去很多线程安全的麻烦,让代码更简洁,运行也更稳定。
希望这个解释清楚了为什么选择ConcurrentHashMap!
聊天应用中的私聊和群聊数据查询优化
在开发聊天应用时,如何查询和展示私聊和群聊的会话列表是一个关键问题。我们需要从服务端向客户端传递两种类型的数据:私聊消息和群聊消息。为了实现这一点,我们使用了ConcurrentHashMap来存储这些数据,确保线程安全和高效的并发访问。

聊天应用中的私聊和群聊数据查询优化

在开发聊天应用时,如何查询和展示私聊和群聊的会话列表是一个关键问题。我们需要从服务端向客户端传递两种类型的数据:私聊消息和群聊消息。为了实现这一点,我们使用了ConcurrentHashMap来存储这些数据,确保线程安全和高效的并发访问。

以下是服务端代码的结构:

ConcurrentHashMap<Long, List<ChatMessage>> messageMap = new ConcurrentHashMap<>();
ConcurrentHashMap<Long, List<GroupMessage>> groupMessageMap = new ConcurrentHashMap<>();

私聊会话查询

私聊会话的查询需要获取以下信息:

  1. 用户头像、用户名和会话是否置顶。
  2. 最后一条消息、最后活动时间和未读消息数量。

查询私聊用户信息

SELECT m.user_id, m.isPinned, u.username, u.image 
FROM friend m 
JOIN user u ON m.user_id = u.id 
WHERE m.friend_id = ?
  • friend:存储用户之间的关系,user_id表示好友的ID,friend_id表示当前用户的ID,isPinned表示是否置顶。
  • user:存储用户的基本信息,username是用户名,image是用户头像。

查询私聊消息信息

SELECT 
    CASE WHEN sender_id = ? THEN receiver_id ELSE sender_id END as chat_id, 
    MAX(time) as last_time,
    (SELECT content FROM messages m 
     WHERE (m.sender_id = ? OR m.receiver_id = ?) 
     AND (CASE WHEN m.sender_id = ? THEN m.receiver_id ELSE m.sender_id END) = chat_id
     AND m.room_id IS NULL
     ORDER BY m.time DESC LIMIT 1) as last_message,
    SUM(CASE WHEN receiver_id = ? AND status = 0 THEN 1 ELSE 0 END) as unread_count
FROM messages 
WHERE (sender_id = ? OR receiver_id = ?) AND room_id IS NULL 
GROUP BY chat_id
ORDER BY last_time DESC;
  • messages:存储消息内容,sender_idreceiver_id分别表示发送者和接收者的ID,time表示消息发送时间,status表示消息的读取状态(0表示未读,1表示已读)。
  • 逻辑解释
    • CASE WHEN sender_id = ? THEN receiver_id ELSE sender_id END as chat_id:根据当前用户ID,确定对方的用户ID。
    • MAX(time):获取最后一条消息的时间。
    • 子查询获取最后一条消息的内容。
    • SUM(CASE WHEN receiver_id = ? AND status = 0 THEN 1 ELSE 0 END):统计未读消息的数量。

群聊会话查询

群聊会话的查询需要获取以下信息:

  1. 群组名称、群组头像、是否置顶。
  2. 最后一条消息、最后活动时间和未读消息数量。

查询群聊信息

SELECT 
    g.id AS group_id,
    g.name AS group_name,
    g.image AS group_avatar,
    MAX(ug.timeship) AS last_active_time,
    (SELECT ug2.message 
     FROM user_group ug2 
     WHERE ug2.group_id = g.id 
     ORDER BY ug2.timeship DESC 
     LIMIT 1) AS last_message,
    SUM(CASE WHEN ug.status = 1 AND ug.user_id != ? THEN 1 ELSE 0 END) AS unread_count,
    MAX(ug.isPinned) AS is_pinned
FROM groupsql g
JOIN user_group ug ON g.id = ug.group_id
WHERE ug.user_id = ? OR 
    EXISTS (SELECT 1 FROM user_group WHERE group_id = g.id AND user_id = ?)
GROUP BY g.id, g.name, g.image
ORDER BY is_pinned DESC, last_active_time DESC;
  • groupsql:存储群组的基本信息,id是群组ID,name是群组名称,image是群组头像。
  • user_group:存储用户与群组的关系,group_id是群组ID,user_id是用户ID,timeship是用户加入群组的时间,message是群组消息,status表示消息的读取状态(1表示未读)。
  • 逻辑解释
    • MAX(ug.timeship):获取群组的最后活动时间。
    • 子查询获取最后一条消息的内容。
    • SUM(CASE WHEN ug.status = 1 AND ug.user_id != ? THEN 1 ELSE 0 END):统计未读消息的数量。
    • MAX(ug.isPinned):判断群组是否置顶。
    • ORDER BY is_pinned DESC, last_active_time DESC:按置顶优先级和最后活动时间排序。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2375169.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

FFmpeg多路节目流复用为一路包含多个节目的输出流

在音视频处理领域&#xff0c;将多个独立的节目流&#xff08;如不同频道的音视频内容&#xff09;合并为一个包含多个节目的输出流是常见需求。FFmpeg 作为功能强大的多媒体处理工具&#xff0c;提供了灵活的流复用能力&#xff0c;本文将通过具体案例解析如何使用 FFmpeg 实现…

分子动力学模拟揭示点突变对 hCFTR NBD1结构域热稳定性的影响

囊性纤维化&#xff08;CF&#xff09; 作为一种严重的常染色体隐性遗传疾病&#xff0c;全球约有 10 万名患者深受其害。它会累及人体多个器官&#xff0c;如肺部、胰腺等&#xff0c;严重影响患者的生活质量和寿命。CF 的 “罪魁祸首” 是 CFTR 氯离子通道的突变&#xff0c;…

关于SIS/DCS点检周期

在中国化工行业&#xff0c;近几年在设备维护上有个挺有意思的现象&#xff0c;即SIS和DCS这两个系统的点检周期问题&#xff0c;隔三差五就被管理层会议讨论&#xff0c;可以说是企业管理层关注的重要方向与关心要素。 与一般工业行业中设备运维不同&#xff0c;SIS与DCS的点…

【PmHub后端篇】PmHub中基于Redis加Lua脚本的计数器算法限流实现

1 限流的重要性 在高并发系统中&#xff0c;保护系统稳定运行的关键技术有缓存、降级和限流。 缓存通过在内存中存储常用数据&#xff0c;减少对数据库的访问&#xff0c;提升系统响应速度&#xff0c;如浏览器缓存、CDN缓存等多种应用层面。降级则是在系统压力过大或部分服务…

CST软件仿真案例——太阳能薄膜频谱吸收率

CST软件中的太阳能薄膜的功率吸收可用光频电磁波在介质材料中的损耗来计算。本案例计算非晶硅的功率吸收&#xff0c;然后考虑真实太阳频谱&#xff0c;计算有效吸收频谱。 用太阳能单元模板&#xff0c;时域求解器&#xff1a; 材料库提取四个材料&#xff0c;非晶硅&#xf…

ABAP+旧数据接管的会计年度未确定

导资产主数据时&#xff0c;报错旧数据接管的会计年度未确定 是因为程序里面使用了下列函数AISCO_CALCULATE_FIRST_DAY&#xff0c;输入公司代码&#xff0c;获取会计年度&#xff0c;这个数据是在后台表T093C表中取数的&#xff0c;通过SE16N可以看到后台表数据没有数&#xf…

养生:打造健康生活的全方位策略

在生活节奏不断加快的当下&#xff0c;养生已成为提升生活质量、维护身心平衡的重要方式。从饮食、运动到睡眠&#xff0c;再到心态调节&#xff0c;各个方面的养生之道共同构建起健康生活的坚实基础。以下为您详细介绍养生的关键要点&#xff0c;助您拥抱健康生活。 饮食养生…

贪吃蛇游戏排行榜模块开发总结:从数据到视觉的实现

一、项目背景与成果概览 在完成贪吃蛇游戏核心玩法后,本次开发重点聚焦于排行榜系统的实现。该系统具备以下核心特性: 🌐 双数据源支持:本地存储(localStorage)与远程API自由切换 🕒 时间维度统计:日榜/周榜/月榜/全时段数据筛选 🎮 模式区分:闯关模式(关卡进度…

屏幕与触摸调试

本章配套视频介绍: 《28-屏幕与触摸设置》 【鲁班猫】28-屏幕与触摸设置_哔哩哔哩_bilibili LubanCat-RK3588系列板卡都支持mipi屏以及hdmi显示屏的显示。 19.1. 旋转触摸屏 参考文章 触摸校准 参考文章 旋转触摸方向 配置触摸旋转方向 1 2 # 1.查看触摸输入设备 xinput…

使用 百度云大模型平台 做 【提示词优化】

1. 百度云大模型平台 百度智能云千帆大模型平台 &#xfeff; 平台功能&#xff1a;演示了阿里云大模型的百炼平台&#xff0c;该平台提供Prompt工程功能&#xff0c;支持在线创建和优化Prompt模板模板类型&#xff1a;平台提供多种预制模板&#xff0c;同时也支持用户自定义…

IJCAI 2025 | 高德首个原生3D生成基座大模型「G3PT」重塑3D生成的未来

国际人工智能联合会议&#xff08;IJCAI&#xff09;是人工智能领域最古老、最具权威性的学术会议之一&#xff0c;自1969年首次举办以来&#xff0c;至今已有近六十年的历史。它见证了人工智能从萌芽到蓬勃发展的全过程&#xff0c;是全球人工智能研究者、学者、工程师和行业专…

Samtec助力电视广播行业

【摘要前言】 现代广播电视技术最有趣的方面之一就是界限的模糊。过去&#xff0c;音频和视频是通过射频电缆传输的模拟技术采集的&#xff0c;而现在&#xff0c;数字世界已经取代了模拟技术。物理胶片和磁带已让位于数字存储设备和流媒体。 在这个过程中&#xff0c;连接器…

密码学--仿射密码

一、实验目的 1、通过实现简单的古典密码算法&#xff0c;理解密码学的相关概念 2、理解明文、密文、加密密钥、解密密钥、加密算法、解密算法、流密码与分组密码等。 二、实验内容 1、题目内容描述 ①随机生成加密密钥&#xff0c;并验证密钥的可行性 ②从plain文件读入待…

SpringBoot整合MQTT实战:基于EMQX实现双向设备通信(附源码)

简言&#xff1a; 在万物互联的时代&#xff0c;MQTT协议凭借其轻量级、高效率的特性&#xff0c;已成为物联网通信的事实标准。本教程将带领您在Ubuntu系统上搭建EMQX 5.9.0消息服务器&#xff0c;并使用Spring Boot快速实现两个客户端的高效通信。通过本指南&#xff0c;您将…

从零开始掌握FreeRTOS(2)链表之节点的定义

目录 节点 节点定义 节点实现 根节点 根节点定义 精简节点定义 根节点实现 在上篇文章,我们完成了 FreeRTOS 的移植。在创建任务之前,我们需要先了解FreeRTOS的运转机制。 FreeRTOS是一个多任务系统,由操作系统来管理执行每个任务。这些任务全都挂载到一个双向循…

【数据结构】——双向链表

一、链表的分类 我们前面学习了单链表&#xff0c;其是我们链表中的其中一种&#xff0c;我们前面的单链表其实全称是单向无头不循环链表&#xff0c;我们的链表从三个维度进行分类&#xff0c;一共分为八种。 1、单向和双向 可以看到第一个链表&#xff0c;其只能找到其后一个…

mybatis中${}和#{}的区别

先测试&#xff0c;再说结论 userService.selectStudentByClssIds(10000, "wzh or 11");List<StudentEntity> selectStudentByClssIds(Param("stuId") int stuId, Param("field") String field);<select id"selectStudentByClssI…

抗量子计算攻击的数据安全体系构建:从理论突破到工程实践

在“端 - 边 - 云”三级智能协同理论中&#xff0c;端 - 边、边 - 云之间要进行数据传输&#xff0c;网络的安全尤为重要&#xff0c;为了实现系统总体的安全可控&#xff0c;将构建安全网络。 可先了解我的前文&#xff1a;“端 - 边 - 云”三级智能协同平台的理论建构与技术实…

uniapp|实现手机通讯录、首字母快捷导航功能、多端兼容(H5、微信小程序、APP)

基于uniapp实现带首字母快捷导航的通讯录功能,通过拼音转换库实现汉字姓名首字母提取与分类,结合uniapp的scroll-view组件与pageScrollTo API完成滚动定位交互,并引入uni-indexed-list插件优化索引栏性能。 目录 核心功能实现动态索引栏生成​联系人列表渲染​滚动定位联动性…

【Linux】基础IO(二)

&#x1f4dd;前言&#xff1a; 上篇文章我们对Linux的基础IO有了一定的了解&#xff0c;这篇文章我们来讲讲IO更底层的东西&#xff1a; 重定向及其原理感受file_operation文件缓冲区 &#x1f3ac;个人简介&#xff1a;努力学习ing &#x1f4cb;个人专栏&#xff1a;Linux…