成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術文章
文章詳情頁

SpringBoot集成WebSocket實現后臺向前端推送信息的示例

瀏覽:6日期:2023-04-01 14:54:05
前言

在一次項目開發中,使用到了Netty網絡應用框架,以及MQTT進行消息數據的收發,這其中需要后臺來將獲取到的消息主動推送給前端,于是就使用到了MQTT,特此記錄一下。

一、什么是websocket?

WebSocket協議是基于TCP的一種新的網絡協議。它實現了客戶端與服務器全雙工通信,學過計算機網絡都知道,既然是全雙工,就說明了服務器可以主動發送信息給客戶端。這與我們的推送技術或者是多人在線聊天的功能不謀而合。

SpringBoot集成WebSocket實現后臺向前端推送信息的示例

為什么不使用HTTP 協議呢?這是因為HTTP是單工通信,通信只能由客戶端發起,客戶端請求一下,服務器處理一下,這就太麻煩了。于是websocket應運而生。

SpringBoot集成WebSocket實現后臺向前端推送信息的示例

下面我們就直接開始使用Springboot開始整合。以下案例都在我自己的電腦上測試成功,你可以根據自己的功能進行修改即可。

我的項目結構如下:

SpringBoot集成WebSocket實現后臺向前端推送信息的示例

二、使用步驟1.添加依賴

Maven依賴:

<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-websocket</artifactId> </dependency> 2.啟用Springboot對WebSocket的支持

啟用WebSocket的支持也是很簡單,幾句代碼搞定:

import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.web.socket.server.standard.ServerEndpointExporter;/** * @ Auther: 馬超偉 * @ Date: 2020/06/16/14:35 * @ Description: 開啟WebSocket支持 */@Configurationpublic class WebSocketConfig { @Bean public ServerEndpointExporter serverEndpointExporter() { return new ServerEndpointExporter(); }}3.核心配置:WebSocketServer

因為WebSocket是類似客戶端服務端的形式(采用ws協議),那么這里的WebSocketServer其實就相當于一個ws協議的Controller

@ ServerEndpoint 注解是一個類層次的注解,它的功能主要是將目前的類定義成一個websocket服務器端, 注解的值將被用于監聽用戶連接的終端訪問URL地址,客戶端可以通過這個URL來連接到WebSocket服務器端 新建一個ConcurrentHashMap webSocketMap 用于接收當前userId的WebSocket,方便傳遞之間對userId進行推送消息。

下面是具體業務代碼:

package cc.mrbird.febs.external.webScoket;import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper;import lombok.extern.slf4j.Slf4j;import org.springframework.stereotype.Component;import org.springframework.stereotype.Service;import javax.websocket.*;import javax.websocket.server.PathParam;import javax.websocket.server.ServerEndpoint;import java.io.IOException;import java.time.LocalDateTime;import java.util.List;import java.util.concurrent.CopyOnWriteArraySet;/** * Created with IntelliJ IDEA. * @ Auther: 馬超偉 * @ Date: 2020/06/16/14:35 * @ Description: * @ ServerEndpoint 注解是一個類層次的注解,它的功能主要是將目前的類定義成一個websocket服務器端, * 注解的值將被用于監聽用戶連接的終端訪問URL地址,客戶端可以通過這個URL來連接到WebSocket服務器端 */@Component@Slf4j@Service@ServerEndpoint('/api/websocket/{sid}')public class WebSocketServer { //靜態變量,用來記錄當前在線連接數。應該把它設計成線程安全的。 private static int onlineCount = 0; //concurrent包的線程安全Set,用來存放每個客戶端對應的MyWebSocket對象。 private static CopyOnWriteArraySet<WebSocketServer> webSocketSet = new CopyOnWriteArraySet<WebSocketServer>(); //與某個客戶端的連接會話,需要通過它來給客戶端發送數據 private Session session; //接收sid private String sid = ''; /** * 連接建立成功調用的方法 */ @OnOpen public void onOpen(Session session, @PathParam('sid') String sid) { this.session = session; webSocketSet.add(this); //加入set中 this.sid = sid; addOnlineCount(); //在線數加1 try { sendMessage('conn_success'); log.info('有新窗口開始監聽:' + sid + ',當前在線人數為:' + getOnlineCount()); } catch (IOException e) { log.error('websocket IO Exception'); } } /** * 連接關閉調用的方法 */ @OnClose public void onClose() { webSocketSet.remove(this); //從set中刪除 subOnlineCount(); //在線數減1 //斷開連接情況下,更新主板占用情況為釋放 log.info('釋放的sid為:'+sid); //這里寫你 釋放的時候,要處理的業務 log.info('有一連接關閉!當前在線人數為' + getOnlineCount()); } /** * 收到客戶端消息后調用的方法 * @ Param message 客戶端發送過來的消息 */ @OnMessage public void onMessage(String message, Session session) { log.info('收到來自窗口' + sid + '的信息:' + message); //群發消息 for (WebSocketServer item : webSocketSet) { try { item.sendMessage(message); } catch (IOException e) { e.printStackTrace(); } } } /** * @ Param session * @ Param error */ @OnError public void onError(Session session, Throwable error) { log.error('發生錯誤'); error.printStackTrace(); } /** * 實現服務器主動推送 */ public void sendMessage(String message) throws IOException { this.session.getBasicRemote().sendText(message); } /** * 群發自定義消息 */ public static void sendInfo(String message, @PathParam('sid') String sid) throws IOException { log.info('推送消息到窗口' + sid + ',推送內容:' + message); for (WebSocketServer item : webSocketSet) { try { //這里可以設定只推送給這個sid的,為null則全部推送 if (sid == null) {// item.sendMessage(message); } else if (item.sid.equals(sid)) { item.sendMessage(message); } } catch (IOException e) { continue; } } } public static synchronized int getOnlineCount() { return onlineCount; } public static synchronized void addOnlineCount() { WebSocketServer.onlineCount++; } public static synchronized void subOnlineCount() { WebSocketServer.onlineCount--; } public static CopyOnWriteArraySet<WebSocketServer> getWebSocketSet() { return webSocketSet; }}4.測試Controller

import org.springframework.stereotype.Controller;import org.springframework.web.bind.annotation.GetMapping;import org.springframework.web.bind.annotation.PathVariable;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.ResponseBody;import org.springframework.web.servlet.ModelAndView;import java.io.IOException;import java.util.HashMap;import java.util.Map;/** * Created with IntelliJ IDEA. * * @ Auther: 馬超偉 * @ Date: 2020/06/16/14:38 * @ Description: */@Controller('web_Scoket_system')@RequestMapping('/api/socket')public class SystemController { //頁面請求 @GetMapping('/index/{userId}') public ModelAndView socket(@PathVariable String userId) { ModelAndView mav = new ModelAndView('/socket1'); mav.addObject('userId', userId); return mav; } //推送數據接口 @ResponseBody @RequestMapping('/socket/push/{cid}') public Map pushToWeb(@PathVariable String cid, String message) { Map<String,Object> result = new HashMap<>(); try { WebSocketServer.sendInfo(message, cid); result.put('code', cid); result.put('msg', message); } catch (IOException e) { e.printStackTrace(); } return result; }}5.測試頁面index.html

<!DOCTYPE html><html><head><meta charset='utf-8'><title>Java后端WebSocket的Tomcat實現</title><script type='text/javascript' src='http://www.piao2010.com/bcjs/js/jquery.min.js'></script></head><body><div style='width: 1200px;height:800px;'></div>Welcome<br/><input type='text' /><button onclick='send()'>發送消息</button><hr/><button onclick='closeWebSocket()'>關閉WebSocket連接</button><hr/><div id='message'></div></body><script type='text/javascript'>var websocket = null;//判斷當前瀏覽器是否支持WebSocketif(’WebSocket’ in window) {//改成你的地址websocket = new WebSocket('ws://192.168.100.196:8082/api/websocket/100');} else {alert(’當前瀏覽器 Not support websocket’)}//連接發生錯誤的回調方法websocket.onerror = function() {setMessageInnerHTML('WebSocket連接發生錯誤');};//連接成功建立的回調方法websocket.onopen = function() {setMessageInnerHTML('WebSocket連接成功');}var U01data, Uidata, Usdata//接收到消息的回調方法websocket.onmessage = function(event) {console.log(event);setMessageInnerHTML(event);setechart()}//連接關閉的回調方法websocket.onclose = function() {setMessageInnerHTML('WebSocket連接關閉');}//監聽窗口關閉事件,當窗口關閉時,主動去關閉websocket連接,防止連接還沒斷開就關閉窗口,server端會拋異常。window.onbeforeunload = function() {closeWebSocket();}//將消息顯示在網頁上function setMessageInnerHTML(innerHTML) {document.getElementById(’message’).innerHTML += innerHTML + ’<br/>’;}//關閉WebSocket連接function closeWebSocket() {websocket.close();}//發送消息function send() {var message = document.getElementById(’text’).value;websocket.send(’{'msg':'’ + message + ’'}’);setMessageInnerHTML(message + '&#13;');}</script></html>6.結果展示

后臺:如果有連接請求

SpringBoot集成WebSocket實現后臺向前端推送信息的示例

前臺顯示:

SpringBoot集成WebSocket實現后臺向前端推送信息的示例

總結

這中間我遇到一個問題,就是說WebSocket啟動的時候優先于spring容器,從而導致在WebSocketServer中調用業務Service會報空指針異常

所以需要在WebSocketServer中將所需要用到的service給靜態初始化一下:如圖所示:

SpringBoot集成WebSocket實現后臺向前端推送信息的示例

還需要做如下配置:

SpringBoot集成WebSocket實現后臺向前端推送信息的示例

到此這篇關于SpringBoot集成WebSocket實現后臺向前端推送信息的示例的文章就介紹到這了,更多相關SpringBoot集成WebSocket 內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Spring
相關文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
日韩av一二三| 欧美片在线播放| 91精品国产aⅴ一区二区| 日韩精品国产精品| 亚洲自拍另类| 一区二区三区四区av| 在线不卡视频| 一区精品在线播放| 欧美三级黄美女| 欧美国产成人在线| 欧美国产三级| 一区精品在线播放| 亚洲高清在线| 一卡二卡欧美日韩| 香蕉亚洲视频| 中文字幕一区二区三区av| 在线不卡视频| 亚洲在线中文字幕| 久久天堂成人| 男女激情视频一区| 欧美写真视频网站| 精品一区二区三区在线观看国产 | 成人一区二区视频| 欧美一区二区在线视频| 成人综合婷婷国产精品久久 | 丰满少妇在线播放bd日韩电影| 欧美一区二区三区四区五区| 国产成人免费xxxxxxxx| 欧美福利视频一区| 不卡的av中国片| 久久男人中文字幕资源站| 欧美不卡三区| 综合在线观看色| 亚洲精品韩国| 亚洲二区视频在线| 日本韩国欧美一区二区三区| 捆绑调教美女网站视频一区| 欧美艳星brazzers| 国产成人欧美日韩在线电影| 26uuu国产一区二区三区| 91麻豆精品在线观看| 国产精品免费丝袜| 中文精品视频一区二区在线观看| 亚洲午夜在线观看视频在线| 在线中文字幕一区| 国产一区二区免费视频| 26uuu另类欧美亚洲曰本| 91论坛在线播放| 国产女人水真多18毛片18精品视频| 欧美午夜免费| 亚洲一区二区三区中文字幕在线| 91福利视频久久久久| 国产精品一区二区x88av| 久久久久国产免费免费 | 国产精品免费区二区三区观看| 日韩电影在线免费看| 91精品欧美一区二区三区综合在| 91性感美女视频| 一区二区三区在线免费观看| 日本乱码高清不卡字幕| 国产成人在线观看免费网站| 国产精品私人影院| 国产三区精品| 精品一区二区三区蜜桃| 久久亚洲二区三区| 一区在线视频观看| 日韩二区三区四区| 日韩欧美成人午夜| 亚洲精品欧洲| 久久精品国产一区二区三区免费看 | 舔着乳尖日韩一区| 欧美一级艳片视频免费观看| 欧美午夜影院| 亚洲欧美日韩成人高清在线一区| 色噜噜狠狠一区二区三区果冻| 国产91精品免费| 综合电影一区二区三区| 老**午夜毛片一区二区三区| 成人午夜视频在线观看| 曰韩精品一区二区| 欧美一级久久久| 在线精品一区| 国产一区二区三区久久久| 国产精品私房写真福利视频| 欧美羞羞免费网站| 欧美日韩国产综合视频在线| 免费在线看成人av| 国产欧美日韩亚州综合| 色视频一区二区| 色综合久久中文综合久久97| 午夜天堂影视香蕉久久| 精品粉嫩超白一线天av| 在线亚洲精品| 国产成人精品免费在线| 亚洲自拍欧美精品| 欧美成人欧美edvon| 久久久蜜桃一区二区人| 91丝袜国产在线播放| 免费的国产精品| 国产精品国产三级国产普通话三级| 在线观看成人小视频| 午夜精品久久久久99热蜜桃导演| 麻豆精品一二三| 国产精品视频一二| 91麻豆精品国产自产在线| 国产精品手机在线| 欧美一区二区| 日韩电影免费一区| 国产精品美女久久久久久久久 | 日本亚洲一区二区| 国产农村妇女毛片精品久久麻豆| 在线视频欧美区| 欧美日韩专区| 国产高清一区日本| 亚洲成a人片综合在线| 国产三级久久久| 久久久久久一区| 欧美精品三区| 国产成人精品亚洲777人妖| 亚洲视频免费观看| 欧美精品一区二| 欧美亚洲国产一区在线观看网站| 亚洲私人影院| 成人黄色av电影| 久久99精品国产| 亚洲国产精品精华液网站| 久久久.com| 欧美日韩国产首页在线观看| 国产视频一区三区| 欧美视频四区| 波多野结衣在线一区| 日本欧美一区二区在线观看| 中文字幕一区二区视频| 久久一日本道色综合| 911精品产国品一二三产区| 另类亚洲自拍| 亚洲日本精品国产第一区| 99国产精品国产精品毛片| 久草热8精品视频在线观看| 亚洲午夜免费电影| 国产精品毛片久久久久久| 日韩精品一区二区三区四区| 欧洲人成人精品| 亚久久调教视频| 亚洲激情啪啪| 国产一区欧美| 99免费精品在线观看| 国产毛片精品视频| 麻豆成人久久精品二区三区红| 亚洲一区二区三区小说| 亚洲男人电影天堂| 亚洲欧洲国产日本综合| 久久蜜桃av一区精品变态类天堂 | 日韩精品成人一区二区在线| 亚洲天堂精品视频| 国产精品网友自拍| 国产精品国产自产拍高清av王其 | 欧美日韩hd| 91麻豆自制传媒国产之光| 高清beeg欧美| 国产精品1区2区| 国产综合久久久久影院| 美女任你摸久久| 亚洲日本一区二区三区| 中文字幕欧美国产| 久久久不卡网国产精品二区| 欧美精品一区二区三区蜜桃| 4438x成人网最大色成网站| 欧美伊人久久久久久久久影院 | 亚洲一区二区三区色| 最新国产拍偷乱拍精品| 国产精品igao视频网网址不卡日韩| 91片在线免费观看| 欧美成人tv| 黄色成人av网站| 欧美日本一区| 你懂的视频一区二区| 欧美一区二区三区另类 | 欧美电影免费观看高清完整版在线观看| 欧美日韩一区二区在线视频| 欧美午夜不卡在线观看免费| 欧美系列亚洲系列| 欧美一区二区网站| 久久这里只有精品首页| 国产亚洲婷婷免费| 国产欧美精品在线观看| 国产精品看片你懂得| 亚洲美女淫视频| 婷婷久久综合九色综合伊人色| 免费成人在线视频观看| 国产在线日韩欧美| 成人性生交大片| 欧美日韩精品免费观看 | 欧美色视频在线| 欧美日本韩国一区二区三区视频| 7777精品伊人久久久大香线蕉| 91精品国产黑色紧身裤美女| 日韩欧美一级精品久久| 久久久高清一区二区三区| 日本一区二区成人|