Springboot整合WebSocket 實現(xiàn)聊天室功能
前言
WebSocket概述:
在日常的web應(yīng)用開發(fā)中,常見的是前端向后端發(fā)起請求,有些時候會涉及到前后端互發(fā)消息,這時候就用到了WebSocket。
一、WebSocket原理
WebSocket是一種在單個TCP連接上進行全雙工通信的協(xié)議。它通過一個簡單的握手過程來建立連接,然后在連接上進行雙向數(shù)據(jù)傳輸。與傳統(tǒng)的HTTP請求不同,WebSocket連接一旦建立,就可以在客戶端和服務(wù)器之間保持打開狀態(tài),直到被任何一方關(guān)閉。
核心特點包括:
- 全雙工通信:客戶端和服務(wù)器可以同時發(fā)送和接收消息。
- 持久連接:一旦建立連接,就可以持續(xù)進行數(shù)據(jù)交換,無需像HTTP那樣頻繁地建立新的連接。
- 低延遲:由于連接是持久的,數(shù)據(jù)可以幾乎實時地發(fā)送和接收。
- 輕量級協(xié)議:WebSocket協(xié)議的頭部信息非常簡單,減少了數(shù)據(jù)傳輸?shù)拈_銷。
二、Spring Boot集成WebSocket
代碼結(jié)構(gòu):

2.1. 引入依賴
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-websocket</artifactId>
</dependency>
2.2 配置類WebSocketConfig
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.socket.server.standard.ServerEndpointExporter;
import org.springframework.web.socket.server.standard.ServletServerContainerFactoryBean;
/**
* WebSocket配置類。
* 用于啟用Spring WebSocket支持,通過@Bean注解注冊ServerEndpointExporter,
* 從而允許使用@ServerEndpoint注解定義WebSocket端點。
*/
@Configuration
public class WebSocketConfig {
/**
* 注冊ServerEndpointExporter Bean。
* ServerEndpointExporter是Spring提供的一個工具類,
* 它會掃描并注冊所有使用@ServerEndpoint注解的類為WebSocket端點。
*
* @return ServerEndpointExporter實例
*/
@Bean
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
/**
* 通信文本消息和二進制緩存區(qū)大小
* 避免報文過大時,Websocket 1009 錯誤
*/
@Bean
public ServletServerContainerFactoryBean createWebSocketContainer() {
ServletServerContainerFactoryBean container = new ServletServerContainerFactoryBean();
// 文本/二進制消息最大緩沖區(qū)(10MB)
container.setMaxTextMessageBufferSize(1024 * 1024 * 10);
container.setMaxBinaryMessageBufferSize(1024 * 1024 * 10);
// 最大會話空閑超時時間(1小時)
container.setMaxSessionIdleTimeout(60 * 60 * 1000L);
return container;
}
}
2.3 WebSocketServer 類
WebSocketServer 類實現(xiàn)了 WebSocket 服務(wù)端的功能。 負責(zé)處理 WebSocket 連接的建立、關(guān)閉、消息接收和發(fā)送等操作。
import lombok.Getter;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
import javax.websocket.*;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;
import java.util.Objects;
import java.util.concurrent.CopyOnWriteArraySet;
/**
* WebSocketServer 類實現(xiàn)了 WebSocket 服務(wù)端的功能。
* 它負責(zé)處理 WebSocket 連接的建立、關(guān)閉、消息接收和發(fā)送等操作。
*/
@Component
@Slf4j
@ServerEndpoint("/api/websocket/{sid}")
public class WebSocketServer {
// 靜態(tài)變量,用于記錄當(dāng)前在線連接數(shù)
private static int onlineCount = 0;
// 存儲所有連接的 WebSocketServer 實例
@Getter
private static final CopyOnWriteArraySet<WebSocketServer> webSocketSet = new CopyOnWriteArraySet<>();
// 當(dāng)前連接的會話對象
private Session session;
// 客戶端唯一標(biāo)識符
private String sid = "";
/**
* 連接建立成功時調(diào)用的方法。
*
* @param session 當(dāng)前連接的會話對象
* @param sid 客戶端唯一標(biāo)識符
*/
@OnOpen
public void onOpen(Session session, @PathParam("sid") String sid) {
this.session = session;
webSocketSet.add(this); // 將當(dāng)前實例加入集合
this.sid = sid;
addOnlineCount(); // 在線數(shù)加1
try {
sendMessage("WebSocket 連接成功"); // 發(fā)送連接成功的消息
log.info("有新窗口開始監(jiān)聽:{},當(dāng)前在線人數(shù)為:{}", sid, getOnlineCount());
} catch (IOException e) {
log.error("websocket IO Exception");
}
}
/**
* 連接關(guān)閉時調(diào)用的方法。
*/
@OnClose
public void onClose() {
webSocketSet.remove(this); // 從集合中移除當(dāng)前實例
subOnlineCount(); // 在線數(shù)減1
log.info("釋放的sid為:{}", sid);
log.info("有一個連接關(guān)閉!當(dāng)前在線人數(shù)為{}", getOnlineCount());
}
/**
* 接收到客戶端消息時調(diào)用的方法。
*
* @param message 客戶端發(fā)送的消息
* @param session 當(dāng)前連接的會話對象
*/
@OnMessage
public void onMessage(String message, Session session) {
log.info("收到來自窗口{}的信息:{}", sid, message);
// 群發(fā)消息
for (WebSocketServer item : webSocketSet) {
if (Objects.equals(item.sid, this.sid)) {
continue;
}
sendMessageToClient(item, message);
}
}
/**
* 實現(xiàn)服務(wù)器主動推送消息的方法,并統(tǒng)一處理異常。
*
* @param client 要推送的客戶端實例
* @param message 要推送的消息
*/
private void sendMessageToClient(WebSocketServer client, String message) {
try {
client.sendMessage(message);
} catch (IOException e) {
log.error("向客戶端 {} 發(fā)送消息時出錯: {}", client.sid, message, e);
}
}
/**
* 群發(fā)自定義消息給指定的客戶端。
*
* @param message 要發(fā)送的消息
* @param sid 客戶端唯一標(biāo)識符,為 null 時發(fā)送給所有客戶端
* @throws IOException 如果發(fā)送消息時發(fā)生 I/O 錯誤
*/
public static void sendInfo(String message, @PathParam("sid") String sid) throws IOException {
log.info("推送消息到窗口" + sid + ",推送內(nèi)容:" + message);
for (WebSocketServer item : webSocketSet) {
try {
if (sid == null) {
// 如果 sid 為 null,則發(fā)送給所有客戶端
item.sendMessage(message);
} else if (item.sid.equals(sid)) {
// 如果 sid 匹配,則只發(fā)送給該客戶端
item.sendMessage(message);
}
} catch (IOException e) {
log.error("向客戶端 {} 發(fā)送消息時出錯: {}", item.sid, message, e);
}
}
}
/**
* 發(fā)生錯誤時調(diào)用的方法。
*
* @param session 當(dāng)前連接的會話對象
* @param error 發(fā)生的錯誤
*/
@OnError
public void onError(Session session, Throwable error) {
log.error("發(fā)生錯誤");
error.printStackTrace();
}
/**
* 實現(xiàn)服務(wù)器主動推送消息的方法。
*
* @param message 要推送的消息
* @throws IOException 如果發(fā)送消息時發(fā)生 I/O 錯誤
*/
public void sendMessage(String message) throws IOException {
this.session.getBasicRemote().sendText(message);
}
/**
* 獲取當(dāng)前在線連接數(shù)。
*
* @return 當(dāng)前在線連接數(shù)
*/
public static synchronized int getOnlineCount() {
return onlineCount;
}
/**
* 增加在線連接數(shù)。
*/
public static synchronized void addOnlineCount() {
WebSocketServer.onlineCount++;
}
/**
* 減少在線連接數(shù)。
*/
public static synchronized void subOnlineCount() {
WebSocketServer.onlineCount--;
}
}
2.4 前端代碼 index.html
<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="UTF-8">
<title>WebSocket 聊天室</title>
<script src="https://autherp.jd.com/js/jquery.js"></script>
<style>
.time {
font-size: 0.8em;
display: block;
margin-bottom: 5px;
}
.user-msg {
background-color: #90EE90;
padding: 8px;
border-radius: 8px;
max-width: 70%;
display: inline-block;
}
.system-msg {
background-color: #D3D3D3;
padding: 8px;
border-radius: 8px;
max-width: 70%;
display: inline-block;
font-size: 0.9em;
}
#message-box {
height: 300px;
overflow-y: auto;
margin-bottom: 10px;
border: 1px solid #ccc;
padding: 10px;
border-radius: 4px;
}
.message-right {
text-align: right;
margin: 10px 0;
}
.message-left {
text-align: left;
margin: 10px 0;
}
.message-center {
text-align: center;
margin: 10px 0;
}
.container {
max-width: 600px;
margin: 20px auto;
padding: 20px;
border: 1px solid #ddd;
border-radius: 8px;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
}
.input-group {
display: flex;
gap: 10px;
}
.input-group input[type="text"] {
flex-grow: 1;
padding: 8px;
border: 1px solid #ccc;
border-radius: 4px;
}
.btn-primary, .btn-danger {
padding: 8px 16px;
border: none;
border-radius: 4px;
cursor: pointer;
}
.btn-primary {
background-color: #007BFF;
color: white;
}
.btn-primary:hover {
background-color: #0056b3;
}
.btn-danger {
background-color: #DC3545;
color: white;
}
.btn-danger:hover {
background-color: #c82333;
}
/* 添加新樣式讓標(biāo)題和按鈕居中 */
.container h2,
.container .btn-danger {
text-align: center;
display: block;
margin-left: auto;
margin-right: auto;
}
/* 為按鈕添加一些外邊距,使其看起來更美觀 */
.container .btn-danger {
margin-top: 10px;
}
</style>
</head>
<body>
<div class="container">
<h2>WebSocket 聊天室</h2>
<!-- 添加顯示 sid 的元素 -->
<p id="sid-display">當(dāng)前用戶 SID: <span id="sid-value"></span></p>
<!-- 消息顯示區(qū)域 -->
<div id="message-box"></div>
<!-- 輸入框與發(fā)送按鈕 -->
<div class="input-group">
<input type="text" id="text" placeholder="請輸入消息..." />
<button class="btn-primary" onclick="send()">發(fā)送</button>
</div>
<hr/>
<!-- 關(guān)閉連接按鈕 -->
<button class="btn-danger" onclick="closeWebSocket()">關(guān)閉 WebSocket 連接</button>
</div>
<script type="text/javascript">
let websocket = '';
// 獲取當(dāng)前頁面 URL 中的 sid 參數(shù)或隨機生成一個
function getSid() {
const urlParams = new URLSearchParams(window.location.search);
return urlParams.get('sid') || Math.floor(1000 + Math.random() * 9000); // 4位數(shù)字
}
const sid = getSid();
const wsUrl = `ws://127.0.0.1:9999/api/websocket/${sid}`;
// 頁面加載完成后更新 sid 顯示
window.onload = function() {
document.getElementById('sid-value').textContent = sid;
};
// 初始化 WebSocket
if ('WebSocket' in window) {
websocket = new WebSocket(wsUrl);
} else {
alert('當(dāng)前瀏覽器不支持 WebSocket');
}
// 連接成功
websocket.onopen = function () {
console.log('WebSocket 連接成功');
};
// 接收消息
websocket.onmessage = function (event) {
addMessage(event.data);
};
// 錯誤處理
websocket.onerror = function () {
console.log('WebSocket 連接發(fā)生錯誤');
};
// 關(guān)閉連接
websocket.onclose = function () {
this.closeWebSocket();
console.log('WebSocket 連接已關(guān)閉');
};
// 頁面關(guān)閉前斷開連接
window.onbeforeunload = function () {
this.closeWebSocket();
};
// 發(fā)送消息
function send() {
let message = document.getElementById('text').value.trim();
if (!message) return;
if (websocket && websocket.readyState === WebSocket.OPEN) {
websocket.send(`{"msg":"${message}","sid":"${sid}", "time": "${new Date().toLocaleTimeString()}"}`);
addMessage(`{"msg":"${message}","sid":"${sid}", "time": "${new Date().toLocaleTimeString()}"}`);
document.getElementById('text').value = '';
} else {
addMessage("WebSocket 連接未建立,請稍后再試。");
}
}
//關(guān)閉WebSocket連接
function closeWebSocket() {
if (websocket) {
websocket.close();
}
}
// 添加消息到聊天區(qū)
function addMessage(content) {
let msgBox = document.getElementById('message-box');
const time = new Date().toLocaleTimeString();
const div = document.createElement('div');
let messageData;
let isSystemMessage = false;
try {
messageData = JSON.parse(content);
} catch (e) {
isSystemMessage = true;
}
if (isSystemMessage) {
div.className = 'message-center';
div.innerHTML = `
<span class="time">${time}</span>
<span class="system-msg"> ${content}</span>
`;
} else {
const isMyMessage = String(messageData.sid) === String(sid);
div.className = isMyMessage ? 'message-right' : 'message-left';
div.innerHTML = `
<span class="time">${messageData.time}</span>
<span class="${isMyMessage ? 'user-msg' : 'system-msg'}">
${isMyMessage ? '' : '用戶#' + messageData.sid + ':'} ${messageData.msg}
</span>
`;
}
msgBox.appendChild(div);
msgBox.scrollTop = msgBox.scrollHeight;
}
</script>
</body>
</html>
2.5 Controller訪問首頁
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
public class TestController {
@RequestMapping("/")
public String index(){
return "index.html";
}
}
打開多個網(wǎng)頁窗口,訪問ip:端口


到此這篇關(guān)于Springboot整合WebSocket 實現(xiàn)聊天室功能的文章就介紹到這了,更多相關(guān)Springboot WebSocket聊天室內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- SpringBoot分布式WebSocket的實現(xiàn)指南
- SpringBoot實現(xiàn)WebSocket通信過程解讀
- 深入淺出SpringBoot WebSocket構(gòu)建實時應(yīng)用全面指南
- 利用SpringBoot與WebSocket實現(xiàn)實時雙向通信功能
- vue+springboot+webtrc+websocket實現(xiàn)雙人音視頻通話會議(最新推薦)
- Springboot使用Websocket的時候調(diào)取IOC管理的Bean報空指針異常問題
- Java?springBoot初步使用websocket的代碼示例
- SpringBoot3整合WebSocket詳細指南
- SpringBoot實現(xiàn)WebSocket的示例代碼
- Spring Boot集成WebSocket項目實戰(zhàn)的示例代碼
相關(guān)文章
解決RestTemplate 請求接收自定義400+ 或500+錯誤
這篇文章主要介紹了解決RestTemplate 請求接收自定義400+ 或500+錯誤,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
springboot集成普羅米修斯(Prometheus)的方法
這篇文章主要介紹了springboot集成普羅米修斯(Prometheus)的方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
10張圖總結(jié)出并發(fā)編程最佳學(xué)習(xí)路線
這篇文章主要介紹了并發(fā)編程的最佳學(xué)習(xí)路線,文中通過圖片介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-08-08
基于 SASL/SCRAM 讓 Kafka 實現(xiàn)動態(tài)授權(quán)認證的方法
在大數(shù)據(jù)處理和分析中?Apache Kafka?已經(jīng)成為了一個核心組件,本文將從零開始部署?ZooKeeper?和?Kafka?并通過配置?SASL/SCRAM?和?ACL(訪問控制列表)來增強?Kafka?的安全性,需要的朋友可以參考下2024-07-07
java用字節(jié)數(shù)組解決FileInputStream讀取漢字出現(xiàn)亂碼問題
這篇文章主要介紹了java用字節(jié)數(shù)組解決FileInputStream讀取漢字出現(xiàn)亂碼問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-05-05

