Spring Boot整合WebSocket和Redis实现直播间在线人数统计功能

在这里插入图片描述

😄 19年之后由于某些原因断更了三年,23年重新扬帆起航,推出更多优质博文,希望大家多多支持~
🌷 古之立大事者,不惟有超世之才,亦必有坚忍不拔之志
🎐 个人CSND主页——Micro麦可乐的博客
🐥《Docker实操教程》专栏以最新的Centos版本为基础进行Docker实操教程,入门到实战
🌺《RabbitMQ》专栏主要介绍使用JAVA开发RabbitMQ的系列教程,从基础知识到项目实战
🌸《设计模式》专栏以实际的生活场景为案例进行讲解,让大家对设计模式有一个更清晰的理解
💕《Jenkins实战》专栏主要介绍Jenkins+Docker的实战教程,让你快速掌握项目CI/CD,是2024年最新的实战教程
🌞《Spring Boot》专栏主要介绍我们日常工作项目中经常应用到的功能以及技巧,代码样例完整
如果文章能够给大家带来一定的帮助!欢迎关注、评论互动~

Spring Boot整合WebSocket和Redis实现直播间在线人数统计功能

  • 前言
  • 简单了解WebSocket和Redis
  • 开发准备
    • 步骤一:添加依赖
    • 步骤二:配置Redis
    • 步骤三:定义WebSocket处理器
    • 步骤五:编写简单的前端页面
  • 开始测试
  • 总结

前言

在现在这个短视频时代,很多企业也投入到了直播的行业,甚至为了打造自己专属私域某些企业也会开发应用自己的直播系统,而在直播应用中,实时显示在线人数是一个非常重要的功能。

这里博主将详细介绍通过结合 Spring BootWebSocketRedis ,我们可以实现一个简单而高效的直播间在线人数统计功能,并提供完整的代码示例。

简单了解WebSocket和Redis

WebSocket是一种在单个TCP连接上进行全双工通信的协议,适用于需要实时数据更新的应用。Redis是一个高性能的键值存储系统,常用于缓存和消息队列。在这里博主将将使用WebSocket来监控用户的连接状态,并使用 Redis 来存储和统计在线人数。

开发准备

步骤一:添加依赖

首先构建我们的 Spring Boot 项目, 引入相关依赖 WebSocketRedis 的依赖

<dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency>
</dependencies>

步骤二:配置Redis

由于Spring Boot自动装配的原理,我们只需要在配置文件设置 Redis 的连接参数,在需要使用Redis的地方注入 RedisTemplate 即可

spring:#redisredis:# 地址host: 127.0.0.1# 端口,默认为6379port: 6379# 数据库索引database: 0# 密码password:# 连接超时时间timeout: 10slettuce:pool:# 连接池中的最小空闲连接min-idle: 5# 连接池中的最大空闲连接max-idle: 8# 连接池的最大数据库连接数max-active: 20# #连接池最大阻塞等待时间(使用负值表示没有限制)max-wait: -1ms

步骤三:定义WebSocket处理器

创建一个 WebSocket理器类WebSocketHandler继承TextWebSocketHandler,用于处理WebSocket消息和连接事件

package com.toher.dockertestproject.live;import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Component;
import org.springframework.web.socket.CloseStatus;
import org.springframework.web.socket.TextMessage;
import org.springframework.web.socket.WebSocketSession;
import org.springframework.web.socket.handler.TextWebSocketHandler;import java.util.Collections;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;@Component
public class WebSocketHandler extends TextWebSocketHandler {//定义redis keyprivate static final String LIVE_ROOM_USER_KEY = "liveRoomUsers";//注入RedisTemplateprivate final StringRedisTemplate redisTemplate;//使用集合存储每个用户WebSocket会话private final Set<WebSocketSession> sessions = Collections.newSetFromMap(new ConcurrentHashMap<>());public WebSocketHandler(StringRedisTemplate redisTemplate) {this.redisTemplate = redisTemplate;}/*** 建立连接后* @param session 连接会话* @throws Exception*/@Overridepublic void afterConnectionEstablished(WebSocketSession session) throws Exception {sessions.add(session);//调用increment方法进行自增操作redisTemplate.opsForValue().increment(LIVE_ROOM_USER_KEY);}@Overrideprotected void handleTextMessage(WebSocketSession session, TextMessage message) throws Exception {String userId = message.getPayload();// 拟获取用户Id 后返回用户信息String userName = "匿名用户";if(userId.equals("1")){userName = "榜一大哥:小明";}if(userId.equals("2")){userName = "榜二大姐:小羊";}session.sendMessage(new TextMessage("user:"+userName));broadcastOnlineCount();}@Overridepublic void afterConnectionClosed(WebSocketSession session, CloseStatus status) throws Exception {//集合删除会话sessions.remove(session);//调用increment方法进行自减操作redisTemplate.opsForValue().decrement(LIVE_ROOM_USER_KEY);broadcastOnlineCount();}private void broadcastOnlineCount() {String count = redisTemplate.opsForValue().get(LIVE_ROOM_USER_KEY);TextMessage message = new TextMessage("count: " + count);for (WebSocketSession session : sessions)try {session.sendMessage(message);} catch (Exception e) {e.printStackTrace();}}
}

步骤五:编写简单的前端页面

创建一个简单的前端页面,URL传递参数用户ID用于模拟后端获取用户信息返回,创建一个id元素用于连接WebSocket并显示在线人数

<!DOCTYPE html>
<html>
<head><title>欢迎来到麦可乐的直播间</title>
</head>
<body><h1>直播间人数: <span id="onlineCount">0</span></h1><h2 id="user"></h2><script type="text/javascript">const params = new URLSearchParams(window.location.search);let socket = new WebSocket("ws://localhost:9090/ws");socket.onmessage = function(event) {//获取后端传递的数据 格式 xxx:xxxlet data = event.data.split(":");if(data[0]=='user'){let user = `欢迎 ${data[1]} 进入我直播间`document.getElementById("user").innerText = user;}else{document.getElementById("onlineCount").innerText = data[1];}};socket.onopen = function(event) {//模拟发送用户ID socket.send(params.get('userId'));console.log("创建WebSocket会话");};socket.onclose = function(event) {console.log("关闭WebSocket会话");};</script>
</body>
</html>

开始测试

将前后端项目运行,打开多个浏览器窗口(不是标签页!不是标签页!不是标签页!),测试访问
在这里插入图片描述

可以看到如上图所示,当新窗口进入了前端地址直播间人数+1,关闭窗口或所在标签页直播间人数-1

总结

通过本文的步骤,我们成功地在Spring Boot项目整合WebSocket和Redis实现了一个直播间在线人数统计功能。这个解决方案不仅能够实时更新在线人数,还能有效地处理高并发场景。
本文的代码主要是演示使用,小伙伴们可以根据自己业务需求进行修改升级。如果本文对您有所帮助,希望 一键三连 给博主一点点鼓励,如果您有任何疑问或建议,请随时留言讨论。


在这里插入图片描述

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

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

相关文章

高精度加法的实现

这是C算法基础-基础算法专栏的第七篇文章&#xff0c;专栏详情请见此处。 引入 在C语言中&#xff0c;int的可存储数据范围是-2147483648~2147483647&#xff0c;long long的可存储数据范围是-9223372036854775808~9223372036854775807&#xff0c;但是如果一些数据比long long…

【TensorFlow深度学习】实现Actor-Critic算法的关键步骤

实现Actor-Critic算法的关键步骤 实现Actor-Critic算法的关键步骤&#xff1a;强化学习中的双剑合璧Actor-Critic算法简介关键实现步骤代码示例&#xff08;使用TensorFlow&#xff09;结语 实现Actor-Critic算法的关键步骤&#xff1a;强化学习中的双剑合璧 在强化学习的广阔…

Qt下调用Snap7库与西门子PLC通信

文章目录 前言一、Snap7源码下载二、Snap7的dll常用函数功能介绍三、Snap7Lib.pri模块的封装四、下载链接总结 前言 本文主要讲述了在Qt下调用Snap7库与西门子PLC进行通信&#xff0c;在这里将Snap7的源码与动态库整合在一起封装了一个自己的Snap7Lib.pri子模块&#xff0c;方…

java-集合的使用 1

Java 中的集合框架&#xff08;Collection Framework&#xff09;是 Java API 的一部分&#xff0c;提供了对各种集合数据结构的支持。集合框架使得 Java 程序员能够方便地处理集合数据&#xff0c;而不必担心底层的实现细节。Java 集合框架包括一系列接口和类&#xff0c;用于…

【Python】读取文件夹中所有excel文件拼接成一个excel表格 的方法

我们平常会遇到下载了一些Excel文件放在一个文件夹下&#xff0c;而这些Excel文件的格式都一样&#xff0c;这时候需要批量这些文件合并成一个excel 文件里。 在Python中&#xff0c;我们可以使用pandas库来读取文件夹中的所有Excel文件&#xff0c;并将它们拼接成一个Excel表…

尝试使用blazor(一)吐槽blazor,未开始之前,先吐为敬

为什么要写一点关于blazor的文章呢?其实是没什么人看的&#xff0c;我知道blazor目前在国内使用的人数&#xff0c;恐怕一辆大巴车都坐不满。非常冷门&#xff0c;我刚用blazor遇到问题&#xff0c;花钱找人解决&#xff0c;找了国内几个著名的平台&#xff0c;几乎没人会blaz…

【docker】centos7配置docker镜像阿里云加速

国内从 DockerHub 拉取镜像有时会遇到困难&#xff0c;由于网络原因&#xff0c;下载一个Docker官方镜像可能会需要很长的时间&#xff0c;甚至下载失败。此时可以配置镜像加速器。Docker 官方和国内很多云服务商都提供了国内加速器服务。 测试了几次阿里云的加速是最快的。 …

Golang | Leetcode Golang题解之第131题分割回文串

题目&#xff1a; 题解&#xff1a; func partition(s string) (ans [][]string) {n : len(s)f : make([][]int8, n)for i : range f {f[i] make([]int8, n)}// 0 表示尚未搜索&#xff0c;1 表示是回文串&#xff0c;-1 表示不是回文串var isPalindrome func(i, j int) int8…

Hive 面试题(五)

1. 简述分区表和分桶表的区别 &#xff1f; 分区表&#xff08;Partitioned Table&#xff09;和分桶表&#xff08;Bucketed Table&#xff09;都是Hive中用于优化查询性能的数据组织方式&#xff0c;但它们在概念和用途上有所不同&#xff1a; 分区表&#xff08;Partition…

安装 JDK 8

安装包 百度网盘 提取码&#xff1a;6666 安装步骤 安装路径不要有中文或者特殊符号如空格等。 双击安装包开始安装。 更改安装路径&#xff1a; 跳出一个页面&#xff0c;安装公共 JRE&#xff1a; 配置环境变量&#xff1a; 配置成功&#xff1a; 去掉自动更新

git子模块

1 子模块管理的关键文件和配置 在 Git 中使用子模块时&#xff0c;Git 会利用几个特殊的文件和配置来管理子模块。以下是涉及子模块管理的关键文件和配置&#xff1a; 1.1 .gitmodules 这是一个文本文件&#xff0c;位于 Git 仓库的根目录下。它记录了子模块的信息&#xff…

AI数据分析:用deepseek根据Excel数据绘制分裂饼形图

工作任务&#xff1a;要绘制下面表格中月活用户占比的分裂饼形图 在deepseek中输入提示词&#xff1a; 你是一个Python编程专家&#xff0c;要完成一个Python脚本编写的任务&#xff0c;具体步骤如下&#xff1a; 读取Excel文件"F:\AI自媒体内容\AI行业数据分析\poetop5…

【LLM】度小满金融大模型技术创新与应用探索

note 从通用大模型到金融大模型金融大模型的训练技术创新金融大模型的评测方法创新金融大模型的应用实践创新总结&#xff1a;金融大模型迭代路径 一、轩辕大模型 二、垂直大模型训练 1. 数据准备 数据质量是模型效果的保障。首先数据要丰富&#xff0c;这是必备的条件。我们…

r语言编程艺术 mobi:深度解析R语言编程的奥秘与魅力

r语言编程艺术 mobi&#xff1a;深度解析R语言编程的奥秘与魅力 在数字化时代的浪潮中&#xff0c;R语言以其强大的数据处理能力和灵活的编程特性&#xff0c;逐渐成为数据分析领域的热门选择。而《r语言编程艺术 mobi》这本书&#xff0c;则为我们揭示了R语言编程的深层次奥秘…

除了springboot你还有哪些国产java web框架可以选择

在Java Web框架领域&#xff0c;除了广泛使用的Spring Boot之外&#xff0c;还有多个国产框架可供选择。这些框架各具特色&#xff0c;旨在提供更轻量、更快速、更易于使用的解决方案。以下是几个推荐的国产Java Web框架介绍&#xff1a; 1. **Solon**&#xff1a;Solon是一个…

MeiliSearch-轻量级且美丽的搜索引擎

MeiliSearch-轻量级且美丽的搜索引擎 MeiliSearch 是一个功能强大、快速、开源、易于使用和部署的搜索引擎。它具有以下特点&#xff1a; 支持中文搜索&#xff1a;MeiliSearch 对中文有良好的支持&#xff0c;不需要额外的配置。高度可定制&#xff1a;搜索和索引都可以高度…

TCP和udp能使用同一个端口通讯吗

TCP和UDP是可以使用同一个端口进行通讯的。这是因为TCP和UDP是两个完全不同的协议&#xff0c;它们工作在传输层&#xff0c;各自维护不同的连接和会话。每个协议都有自己的端口号空间&#xff0c;因此TCP和UDP可以互不干扰地使用相同的端口号。 但是&#xff0c;需要注意的是…

UML实现图-组件图

概述 组件图(ComponentDiagram)描述了软件的各种组件和它们之间的依赖关系。组件图中通常包含4种元素:组件、程序、包、任务&#xff0c;各个组件之间还可以相互依赖。 一、组件的表示法 组件是定义了良好接口的物理实现单元&#xff0c;是系统中可替换的物理部件。在一般情…

攻防世界---misc---小小的PDF

1、题目描述&#xff0c;下载附件是一个PDF&#xff0c;打开之后是这样&#xff0c;有两页PDF 2、用winhex分析&#xff0c;没有发现奇怪的地方 3、在kali中binwalk发现有多张照片 4、接着使用foremost将图片分离出来&#xff0c; 5、得到3张图片&#xff0c;打开第3张图片&am…

Android音频架构

Android音频架构 前面《Android音频API》介绍了Android系统提供的四个层面的音频API&#xff1a; Java层MediaRecorder&MediaPlayer系列&#xff1b;Java层AudioTrack&AudioRecorder系列&#xff1b;Jni层opensles&#xff1b;JNI层AAudio&#xff08;Android O引入&…