使用 Spring Boot 搭建 WebSocket 服务器实现多客户端连接

在 Web 开发中,WebSocket 为客户端和服务端之间提供了实时双向通信的能力。本篇博客介绍如何使用 Spring Boot 快速搭建一个 WebSocket 服务器,并支持多客户端的连接和消息广播。

1. WebSocket 简介

WebSocket 是 HTML5 的一种协议,提供了客户端和服务器之间的全双工通信。通过 WebSocket,客户端可以与服务器进行持续连接,不用反复建立 HTTP 请求,从而降低延迟,提升通信效率。

为什么选择 Spring Boot 实现 WebSocket?

Spring Boot 简化了 WebSocket 服务器的配置与实现,使我们可以更专注于业务逻辑开发,且配合 @ServerEndpoint 注解实现更加清晰。

2. 项目环境与依赖配置

项目环境

  • Java 版本:JDK 8+
  • Spring Boot 版本:2.x
  • WebSocket 依赖spring-boot-starter-websocket

Maven 依赖

pom.xml 文件中添加 WebSocket 的依赖:

        <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId></dependency>

3. WebSocket 服务端代码实现

在 Spring Boot 中,我们可以使用 @ServerEndpoint 注解创建 WebSocket 服务器端。以下是一个支持多客户端连接的 WebSocket 实现。

WebSocket 服务端代码解析

创建 MultiClientWebSocket 类,并实现以下功能:

  1. 记录当前在线客户端数
  2. 支持客户端连接、断开、消息接收、群发等功能
  3. 通过 @OnOpen@OnMessage@OnClose@OnError 注解处理连接的各个生命周期

完整代码如下:

import lombok.extern.slf4j.Slf4j;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;
import org.springframework.web.socket.server.standard.ServerEndpointExporter;
import javax.websocket.*;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;/*** WebSocketConfig*/
@Component
@ServerEndpoint("/ws/multi")
@Slf4j
public class MultiClientWebSocket {@Beanpublic ServerEndpointExporter serverEndpointExporter() {log.info("WebSocketConfig: serverEndpointExporter init WebSocketConfig 注入完成");return new ServerEndpointExporter();}/*** 记录当前在线连接数*/private static final AtomicInteger onlineCount = new AtomicInteger(0);/*** 存放所有在线的客户端*/private static final Map<String, Session> onlineClients = new ConcurrentHashMap<>();@OnOpenpublic void onOpen(Session session) {onlineCount.incrementAndGet();onlineClients.put(session.getId(), session);log.info("有新连接加入:{},当前在线客户端数为:{}", session.getId(), onlineCount.get());}@OnMessagepublic void onMessage(String message, Session session) throws IOException {System.out.println("收到客户端消息:" + message);session.getBasicRemote().sendText("服务器收到消息:" + message);}@OnClosepublic void onClose(Session session) {onlineCount.decrementAndGet(); // 在线数减1onlineClients.remove(session.getId());log.info("有一连接关闭:{},当前在线客户端数为:{}", session.getId(), onlineCount.get());}@OnErrorpublic void onError(Throwable t) {log.error("WebSocket 连接出错{}", t.getMessage());}/*** 群发消息** @param message 消息内容*/public void sendMessage(String message) {//log.info("开始给在线的客户端{}群发消息{}",onlineClients,message);for (Map.Entry<String, Session> sessionEntry : onlineClients.entrySet()) {Session toSession = sessionEntry.getValue();log.info("服务端给客户端[{}]发送消息{}", toSession.getId(), message);toSession.getAsyncRemote().sendText(message);}}}

代码解析

  • ServerEndpointExporter:Spring Boot 的 WebSocket 配置类,自动注册 @ServerEndpoint 注解声明的 WebSocket。
  • 在线客户端管理:使用 ConcurrentHashMap 存储在线客户端的 Session,通过 AtomicInteger 记录当前连接数。
  • 生命周期注解
    • @OnOpen:当客户端连接时触发,增加在线人数。
    • @OnMessage:当收到消息时触发,服务端处理消息。
    • @OnClose:当连接关闭时触发,减少在线人数。
    • @OnError:当连接出错时触发,记录错误日志。
  • 群发消息sendMessage 方法遍历所有在线客户端的 Session,实现消息广播。

4. 启动与测试

启动 Spring Boot 项目后,WebSocket 服务端地址为:ws://localhost:8080/ws/multi。可以使用 WebSocket 测试工具(例如 Apifox或浏览器控制台)测试 WebSocket 通信。

测试步骤

  1. 连接 WebSocket:客户端连接至 ws://localhost:8080/ws/multi
  2. 发送消息:客户端发送消息,服务端接收到消息后回复。
  3. 断开连接:客户端断开连接,服务端记录连接关闭信息。
import cn.hutool.json.JSONUtil;
import com.google.common.collect.Lists;
import com.lps.config.MultiClientWebSocket;
import com.lps.domain.R;
import lombok.RequiredArgsConstructor;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import java.util.List;/*** @ClassName: WebSocketControl* @Description:* @Author: liu* @Date: 2024-10-30* @Version: 1.0**/
@RestController
@RequestMapping("/webSocket")
@RequiredArgsConstructor
public class WebSocketControl {private final MultiClientWebSocket multiClientWebSocket;@GetMapping("/sendList")public R<String> sad() {//Guava依赖List<String> strList = Lists.newArrayList("发送", "WebSocket消息","测试");String jsonStr = JSONUtil.toJsonStr(strList);multiClientWebSocket.sendMessage(jsonStr);return R.ok(jsonStr);}
}

前端测试demo

<!DOCTYPE html>
<html>
<head><meta charset="UTF-8"><title>WebSocket Demo</title><style>#status {margin-bottom: 10px;}</style><script>let ws;function connectWebSocket() {// 检查 WebSocket 是否已连接if (ws && ws.readyState === WebSocket.OPEN) {console.log('WebSocket 已连接,无需重复连接。');return;}ws = new WebSocket('ws://localhost:8080/ws/multi');ws.onopen = function() {console.log('WebSocket 连接已经建立。');document.getElementById('status').innerText = 'WebSocket 连接已建立。';ws.send('Hello, server!');};ws.onmessage = function(event) {console.log('收到服务器消息:', event.data);document.getElementById('messages').innerText += '收到消息:' + event.data + '\n';};ws.onerror = function(event) {console.error('WebSocket 连接出现错误:', event);document.getElementById('status').innerText = 'WebSocket 连接出现错误。';};ws.onclose = function() {console.log('WebSocket 连接已经关闭。');document.getElementById('status').innerText = 'WebSocket 连接已关闭。';ws = null; // 连接关闭后将 ws 设置为 null,以便重新连接};}function disconnectWebSocket() {if (ws) {ws.close();}}</script>
</head>
<body><h1>WebSocket Demo</h1><div id="status">WebSocket 连接状态</div><button onclick="connectWebSocket()">连接 WebSocket</button><button onclick="disconnectWebSocket()">断开 WebSocket</button><pre id="messages"></pre>
</body>
</html>

5. 总结

通过以上代码示例,我们可以实现一个简单的 WebSocket 服务端,支持多客户端连接和消息广播。此 WebSocket 服务端适用于需要实时消息推送的应用场景,比如聊天室、实时通知系统等。

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

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

相关文章

详细分析Java8中的Duration时间类基本知识(附Demo)

目录 前言1. 基本知识2. Demo3. 拓展Demo 前言 基本的Java知识推荐阅读&#xff1a; java框架 零基础从入门到精通的学习路线 附开源项目面经等&#xff08;超全&#xff09;【Java项目】实战CRUD的功能整理&#xff08;持续更新&#xff09; 1. 基本知识 Duration 类是 Ja…

Day14买卖股票的最佳时机

给定一个数组 prices &#xff0c;它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。 你只能选择 某一天 买入这只股票&#xff0c;并选择在 未来的某一个不同的日子 卖出该股票。设计一个算法来计算你所能获取的最大利润。 返回你可以从这笔交易中获取的最大利润。如…

【星闪EBM-H63开发板】小熊派的开发板实物

前几天介绍了&#xff0c;我参加了小熊派的星闪EBM-H63开发板内测活动。今天去快递站取回来开发板&#xff0c;让我们看看实物照片。 下面是开发板的正面照片&#xff0c;绿色的部分是核心板&#xff0c;蓝色的部分实际上是个底板&#xff0c;引出了串口的管脚&#xff0c;再加…

zookeeper的选举机制

zk中有两种角色&#xff1a;Leader 和 Fllower Leader是自己的集群各台电脑投票选举出来的。 事务&#xff1a;一通操作&#xff0c;要么同时成立&#xff0c;要么都不成立。 举例&#xff1a;Jack 和 Rose Rose 给 Jack&#xff08;小李子&#xff09; 转钱 Rose -100 J…

六:java 基础知识(6)-- Maven项目管理

目录 1. 引言 1.1 什么是 Maven 1.2 Maven 的优势 2. Maven 基础 2.1 Maven 的基本概念 2.2 Maven 的工作原理 2.3 Maven 的安装与配置 3. Maven 项目结构 3.1 Maven 项目的标准目录结构 3.2 POM 文件&#xff08;pom.xml&#xff09;概述 3.3 POM 文件的基本配置 4…

Centrifuge 5425R - 微量离心机技术数据详情——艾本德

离心机&#xff0c;作为实验室中的关键设备&#xff0c;扮演着分离、浓缩和纯化样品的重要角色。而在众多离心机型号中&#xff0c;Centrifuge 5425R 以其出色的性能和多功能性&#xff0c;成为了众多科研人员的首选。下面&#xff0c;我们就来详细了解一下这款微量离心机的技术…

【06】A-Maven项目SVN设置忽略文件

做Web项目开发时&#xff0c;运用的是Maven管理工具对项目进行管理&#xff0c;在项目构建的过程中自动生成了很多不需要SVN进行管理的文件&#xff0c;SVN在对源码进行版本管理时&#xff0c;需要将其忽略&#xff0c;本文给出了具体解决方案。 SVN设置忽略Maven项目中自动生成…

TDengine 签约蘑菇物联,改造通用设备工业互联网平台

在当前工业互联网迅猛发展的背景下&#xff0c;企业面临着日益增长的数据处理需求和智能化转型的挑战。通用工业设备的高能耗问题愈发突出&#xff0c;尤其是由这些设备组成的公辅能源车间&#xff0c;亟需更高效的解决方案来提升设备运行效率&#xff0c;降低能源消耗。为此&a…

革新汽车装配产线:MR30分布式IO模块引领智能制造新时代

在日新月异的汽车制造行业中&#xff0c;每一分每一秒的效率提升都意味着成本的降低与市场竞争力的增强。随着工业4.0时代的到来&#xff0c;智能化、自动化已成为汽车产线升级转型的关键词。在这场技术革命的浪潮中&#xff0c;MR30分布式IO模块以其高效、灵活、可靠的特点&am…

Qt第三课 ----------容器类控件

作者前言 &#x1f382; ✨✨✨✨✨✨&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f382; ​&#x1f382; 作者介绍&#xff1a; &#x1f382;&#x1f382; &#x1f382; &#x1f389;&#x1f389;&#x1f389…

CodeQL学习笔记(4)-CodeQL for Java(程序元素)

最近在学习CodeQL&#xff0c;对于CodeQL就不介绍了&#xff0c;目前网上一搜一大把。本系列是学习CodeQL的个人学习笔记&#xff0c;根据个人知识库笔记修改整理而来的&#xff0c;分享出来共同学习。个人觉得QL的语法比较反人类&#xff0c;至少与目前主流的这些OOP语言相比&…

动态规划28:376. 摆动序列

动态规划解题步骤&#xff1a; 1.确定状态表示&#xff1a;dp[i]是什么 2.确定状态转移方程&#xff1a;dp[i]等于什么 3.初始化&#xff1a;确保状态转移方程不越界 4.确定填表顺序&#xff1a;根据状态转移方程即可确定填表顺序 5.确定返回值 题目链接&#xff1a;376.…

【zlm】h264 vp9 尝试研究

目录 编译与使用libvpx 打包lib 解决方案一 libvpx直接引用 IVF格式 编译libvpx windows下编译libvpx 参考文章 编译与使用libvpx 我们用最新的&#xff1a; x86_64-win64-vs16 最简单的视频编码器&#xff1a;编译&#xff08;libx264&#xff0c;libx265&#xff…

顺序表专题

目录 0. 什么是数据结构&#xff1f; 0. 为什么需要数据结构&#xff1f; 1.顺序表的概念及结构 2.顺序表分类&#xff1a; 3.动态顺序表的实现 4. 顺序表的应用 5. 顺序表的问题及思考 0. 什么是数据结构&#xff1f; 数据结构是由“数据”和“结构”两词结合而来 什…

关于使用svgIcon 菜单折叠 显示文字情况

使用的工具&#xff1a;vue2&#xff0c;ant design vue 问题&#xff1a; **解决&#xff1a;在<svg-icon> 外面包一层 <a-icon> ** 使用: 在 main.js 中&#xff1a;

【JAVA毕业设计】基于Vue和SpringBoot的师生健康管理系统

博主说明&#xff1a;本文项目编号 T 052 &#xff0c;文末自助获取源码 \color{red}{T052&#xff0c;文末自助获取源码} T052&#xff0c;文末自助获取源码 目录 一、系统介绍二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景5.2 国内外研究现状5.3 可行性分析…

双向链表专题

双向链表 1. 双向链表的定义和结构2. 双向链表的实现2.1 结构声明2.2 双向链表的初始化2.3 双向链表的打印2.4 尾插2.5 头插2.6 在指定位置之前插入2.7 在指定位置之后插入数据2.8 尾删2.9 头删2.10 删除指定位置的节点2.11 查找2.12 链表的销毁 3. 双向链表的细节 &#x1f52…

发票真伪查验方式-python数电票批量查验接口、发票ocr文字识别提取

在当今的商业环境中&#xff0c;确保交易的安全性和透明度是每个企业追求的目标。随着电子商务的迅猛发展&#xff0c;发票管理成为了企业财务管理中不可或缺的一环。面对海量的电子发票&#xff0c;企业财务也无需惊慌&#xff0c;翔云发票查验API接口&#xff0c;可以为企业提…

html+js+css实现拖拽式便签留言

前些日子在网上冲浪时&#xff0c;看到一个便签式留言墙&#xff0c;让人耳目一新。心想这个看着不错&#xff0c;额想要。于是便开始搜寻是否有相应开源插件&#xff0c;想将其引入自己的博客中。但是搜寻了一圈&#xff0c;都没有符合预期的,要么功能不符合。有的功能符合&am…

初识网络编程TCP/IP

目录 前言相关名词解释应用层协议——HTTP传输层协议socketTCP帧头格式三次握手、四次挥手 UDPTCP的socket实现 参考博文 前言 刚碰到网络编程&#xff0c;会出现一堆协议、概念、这层次那技术的&#xff0c;头都大了&#xff0c;还是得总结总结…… 相关名词解释 ✨✨网络…