日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例

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

在一次項(xiàng)目開(kāi)發(fā)中,使用到了Netty網(wǎng)絡(luò)應(yīng)用框架,以及MQTT進(jìn)行消息數(shù)據(jù)的收發(fā),這其中需要后臺(tái)來(lái)將獲取到的消息主動(dòng)推送給前端,于是就使用到了MQTT,特此記錄一下。

一、什么是websocket?

WebSocket協(xié)議是基于TCP的一種新的網(wǎng)絡(luò)協(xié)議。它實(shí)現(xiàn)了客戶端與服務(wù)器全雙工通信,學(xué)過(guò)計(jì)算機(jī)網(wǎng)絡(luò)都知道,既然是全雙工,就說(shuō)明了服務(wù)器可以主動(dòng)發(fā)送信息給客戶端。這與我們的推送技術(shù)或者是多人在線聊天的功能不謀而合。

SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例

為什么不使用HTTP 協(xié)議呢?這是因?yàn)镠TTP是單工通信,通信只能由客戶端發(fā)起,客戶端請(qǐng)求一下,服務(wù)器處理一下,這就太麻煩了。于是websocket應(yīng)運(yùn)而生。

SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例

下面我們就直接開(kāi)始使用Springboot開(kāi)始整合。以下案例都在我自己的電腦上測(cè)試成功,你可以根據(jù)自己的功能進(jìn)行修改即可。

我的項(xiàng)目結(jié)構(gòu)如下:

SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例

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

Maven依賴:

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

啟用WebSocket的支持也是很簡(jiǎn)單,幾句代碼搞定:

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: 開(kāi)啟WebSocket支持 */@Configurationpublic class WebSocketConfig { @Bean public ServerEndpointExporter serverEndpointExporter() { return new ServerEndpointExporter(); }}3.核心配置:WebSocketServer

因?yàn)閃ebSocket是類似客戶端服務(wù)端的形式(采用ws協(xié)議),那么這里的WebSocketServer其實(shí)就相當(dāng)于一個(gè)ws協(xié)議的Controller

@ ServerEndpoint 注解是一個(gè)類層次的注解,它的功能主要是將目前的類定義成一個(gè)websocket服務(wù)器端, 注解的值將被用于監(jiān)聽(tīng)用戶連接的終端訪問(wèn)URL地址,客戶端可以通過(guò)這個(gè)URL來(lái)連接到WebSocket服務(wù)器端 新建一個(gè)ConcurrentHashMap webSocketMap 用于接收當(dāng)前userId的WebSocket,方便傳遞之間對(duì)userId進(jìn)行推送消息。

下面是具體業(yè)務(wù)代碼:

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 注解是一個(gè)類層次的注解,它的功能主要是將目前的類定義成一個(gè)websocket服務(wù)器端, * 注解的值將被用于監(jiān)聽(tīng)用戶連接的終端訪問(wèn)URL地址,客戶端可以通過(guò)這個(gè)URL來(lái)連接到WebSocket服務(wù)器端 */@Component@Slf4j@Service@ServerEndpoint('/api/websocket/{sid}')public class WebSocketServer { //靜態(tài)變量,用來(lái)記錄當(dāng)前在線連接數(shù)。應(yīng)該把它設(shè)計(jì)成線程安全的。 private static int onlineCount = 0; //concurrent包的線程安全Set,用來(lái)存放每個(gè)客戶端對(duì)應(yīng)的MyWebSocket對(duì)象。 private static CopyOnWriteArraySet<WebSocketServer> webSocketSet = new CopyOnWriteArraySet<WebSocketServer>(); //與某個(gè)客戶端的連接會(huì)話,需要通過(guò)它來(lái)給客戶端發(fā)送數(shù)據(jù) private Session session; //接收sid private String sid = ''; /** * 連接建立成功調(diào)用的方法 */ @OnOpen public void onOpen(Session session, @PathParam('sid') String sid) { this.session = session; webSocketSet.add(this); //加入set中 this.sid = sid; addOnlineCount(); //在線數(shù)加1 try { sendMessage('conn_success'); log.info('有新窗口開(kāi)始監(jiān)聽(tīng):' + sid + ',當(dāng)前在線人數(shù)為:' + getOnlineCount()); } catch (IOException e) { log.error('websocket IO Exception'); } } /** * 連接關(guān)閉調(diào)用的方法 */ @OnClose public void onClose() { webSocketSet.remove(this); //從set中刪除 subOnlineCount(); //在線數(shù)減1 //斷開(kāi)連接情況下,更新主板占用情況為釋放 log.info('釋放的sid為:'+sid); //這里寫你 釋放的時(shí)候,要處理的業(yè)務(wù) log.info('有一連接關(guān)閉!當(dāng)前在線人數(shù)為' + getOnlineCount()); } /** * 收到客戶端消息后調(diào)用的方法 * @ Param message 客戶端發(fā)送過(guò)來(lái)的消息 */ @OnMessage public void onMessage(String message, Session session) { log.info('收到來(lái)自窗口' + sid + '的信息:' + message); //群發(fā)消息 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('發(fā)生錯(cuò)誤'); error.printStackTrace(); } /** * 實(shí)現(xiàn)服務(wù)器主動(dòng)推送 */ public void sendMessage(String message) throws IOException { this.session.getBasicRemote().sendText(message); } /** * 群發(fā)自定義消息 */ public static void sendInfo(String message, @PathParam('sid') String sid) throws IOException { log.info('推送消息到窗口' + sid + ',推送內(nèi)容:' + message); for (WebSocketServer item : webSocketSet) { try { //這里可以設(shè)定只推送給這個(gè)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.測(cè)試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 { //頁(yè)面請(qǐng)求 @GetMapping('/index/{userId}') public ModelAndView socket(@PathVariable String userId) { ModelAndView mav = new ModelAndView('/socket1'); mav.addObject('userId', userId); return mav; } //推送數(shù)據(jù)接口 @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.測(cè)試頁(yè)面index.html

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

后臺(tái):如果有連接請(qǐng)求

SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例

前臺(tái)顯示:

SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例

總結(jié)

這中間我遇到一個(gè)問(wèn)題,就是說(shuō)WebSocket啟動(dòng)的時(shí)候優(yōu)先于spring容器,從而導(dǎo)致在WebSocketServer中調(diào)用業(yè)務(wù)Service會(huì)報(bào)空指針異常

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

SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例

還需要做如下配置:

SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例

到此這篇關(guān)于SpringBoot集成WebSocket實(shí)現(xiàn)后臺(tái)向前端推送信息的示例的文章就介紹到這了,更多相關(guān)SpringBoot集成WebSocket 內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Spring
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲激情精品| 国产伦久视频在线观看| 都市激情国产精品| 国产一区二区久久久久| 美女免费视频一区| 精品久久久网| 国产精品高颜值在线观看| 国产成人精选| 久久男人av资源站| 高清一区二区| 午夜影院一区| 麻豆91小视频| 日韩中文在线电影| 精品中国亚洲| 高清久久精品| 日韩欧美一区二区三区在线视频 | 高清久久精品| 激情久久久久久| 亚洲激情偷拍| 综合激情婷婷| 麻豆精品在线| 免费观看亚洲| 最新日韩欧美| 一区二区国产在线| 国产精品密蕾丝视频下载| 另类欧美日韩国产在线| 精品视频99| 欧美丰满日韩| 欧美肉体xxxx裸体137大胆| 性色av一区二区怡红| 深夜福利亚洲| 国产一区国产二区国产三区| 中文字幕一区久| 99riav国产精品| 日韩av一区二区三区| 麻豆免费精品视频| 欧美日韩精品一本二本三本 | 久久国产精品免费一区二区三区| 欧美一级二区| 国产精品久久久久9999高清| 国产精品**亚洲精品| 久久久久久久欧美精品| 日韩中文字幕不卡| 嫩呦国产一区二区三区av| 国产白浆在线免费观看| 国产精品呻吟| 青青青国产精品| 久久精品国产精品亚洲毛片| 91精品一区二区三区综合在线爱| 欧美专区18| 欧美日韩一区二区国产| yellow在线观看网址| 亚洲深夜影院| 免费亚洲一区| 国产视频一区三区| 欧美成人一二区| 精品欧美激情在线观看| 欧美一区免费| 婷婷综合社区| 国产精品对白| 黄色在线一区| 精品72久久久久中文字幕| 日韩精品一区二区三区免费观影| 亚洲1区在线观看| 日本不良网站在线观看| 日本亚洲欧洲无免费码在线| 美日韩一区二区三区| 国产精品日韩| 国产69精品久久| 日韩精品亚洲专区在线观看| 日韩成人亚洲| 国产三级精品三级在线观看国产| 视频二区不卡| 国产精品大片免费观看| 三级欧美韩日大片在线看| 中文在线а√天堂| 国产午夜久久av| 黄色精品网站| 免费看av不卡| 国产乱码精品一区二区亚洲| 亚洲小说欧美另类婷婷| 欧美国产不卡| 综合激情网...| 久久免费大视频| 免费在线播放第一区高清av| 麻豆精品网站| 欧美+日本+国产+在线a∨观看| 欧美精品99| 亚欧成人精品| 国产一级一区二区| 伊伊综合在线| 久久免费影院| 日本一区二区中文字幕| 91久久国产| 欧美日韩免费观看视频| 青青伊人久久| 亚洲色图网站| 欧美女激情福利| 99久久99久久精品国产片果冰| 久久精品资源| 国产精品一区毛片| 天海翼精品一区二区三区| 亚洲一区二区动漫| 在线成人动漫av| 97视频热人人精品免费| 国产精品主播| 日本成人中文字幕在线视频| 午夜国产精品视频免费体验区| 久草免费在线视频| 精品精品国产三级a∨在线| 国产精品视频3p| 欧美日韩午夜| 日本欧美在线| 日本色综合中文字幕| 亚洲精品在线二区| 水野朝阳av一区二区三区| 宅男在线一区| 久久精品国产99久久| 成人久久一区| 久久婷婷av| 精精国产xxxx视频在线野外| 91av亚洲| 激情综合五月| 国内在线观看一区二区三区 | 日韩av在线中文字幕| 欧美91在线| 麻豆精品一区二区综合av| 欧美国产极品| 久久99蜜桃| 精品一区二区三区免费看| 老司机精品视频在线播放| 麻豆国产91在线播放| 精品一区电影| 成人精品国产亚洲| 成年男女免费视频网站不卡| 日韩伦理在线一区| 久久国产成人午夜av影院宅| 欧美日韩亚洲在线观看| 中文国产一区| 亚洲精品精选| 久久精品凹凸全集| 国产精品jk白丝蜜臀av小说| 美女国产精品久久久| 色爱综合网欧美| 日韩av首页| 夜夜嗨av一区二区三区网站四季av| 在线国产一区二区| 亚洲男人在线| 日本欧美韩国一区三区| 国产精品对白久久久久粗| 久久精品国产99国产精品| 日韩免费福利视频| 午夜在线精品偷拍| 日韩不卡一区二区三区| 欧美黄色一区| 一区二区小说| 一区二区三区四区日韩| 欧美日韩伊人| 亚洲国产成人二区| 日韩视频一区| 欧美日韩视频免费看| 国产成人精品一区二区三区免费| 欧美二三四区| 亚洲一区不卡| 国产精品一区二区三区美女 | 色婷婷色综合| 午夜久久免费观看| 日韩三级一区| 激情综合五月| 亚洲少妇诱惑| 国产精品www994| 久久久影院免费| 日韩激情网站| 日韩电影二区| 亚洲综合中文| 精品国产91| 亚洲资源av| 国产精品1区| 欧美一区二区性| 欧美日韩精品一区二区三区在线观看| 久久伊人国产| 亚洲一区中文| 久草精品视频| 天堂av在线一区| 成人在线黄色| 日韩中文字幕在线一区| 福利一区视频| 五月激激激综合网色播 | 麻豆精品在线观看| 亚洲一区中文| 在线看片福利| 亚洲日本在线观看视频| 亚洲黄色网址| 欧美亚洲自偷自偷| 欧美日韩视频| 成人一区不卡| 日本va欧美va精品发布| av一区二区高清| 精品一二三区| 亚洲精品高潮|