Spring Boot中使用Server-Sent Events (SSE) 實現(xiàn)實時數(shù)據(jù)推送教程
一、簡介
Server-Sent Events (SSE) 是HTML5引入的一種輕量級的服務器向瀏覽器客戶端單向推送實時數(shù)據(jù)的技術(shù)。在Spring Boot框架中,我們可以很容易地集成并利用SSE來實現(xiàn)實時通信。
二、依賴添加
在Spring Boot項目中,無需額外引入特定的依賴,因為Spring Web MVC模塊已經(jīng)內(nèi)置了對SSE的支持。
輔助Maven
<!-- 集成beetl --> <dependency> <groupId>com.ibeetl</groupId> <artifactId>beetl-framework-starter</artifactId> <version>1.2.30.RELEASE</version> </dependency> <!-- 集成hutool工具類簡便操作 --> <dependency> <groupId>cn.hutool</groupId> <artifactId>hutool-all</artifactId> <version>5.3.10</version> </dependency>
三、編寫核心SSE Client
@Slf4j @Component public class SseClient { private static final Map<String, SseEmitter> sseEmitterMap = new ConcurrentHashMap<>(); /** * 創(chuàng)建連接 */ public SseEmitter createSse(String uid) { //默認30秒超時,設置為0L則永不超時 SseEmitter sseEmitter = new SseEmitter(0l); //完成后回調(diào) sseEmitter.onCompletion(() -> { log.info("[{}]結(jié)束連接...................", uid); sseEmitterMap.remove(uid); }); //超時回調(diào) sseEmitter.onTimeout(() -> { log.info("[{}]連接超時...................", uid); }); //異?;卣{(diào) sseEmitter.onError( throwable -> { try { log.info("[{}]連接異常,{}", uid, throwable.toString()); sseEmitter.send(SseEmitter.event() .id(uid) .name("發(fā)生異常!") .data("發(fā)生異常請重試!") .reconnectTime(3000)); sseEmitterMap.put(uid, sseEmitter); } catch (IOException e) { e.printStackTrace(); } } ); try { sseEmitter.send(SseEmitter.event().reconnectTime(5000)); } catch (IOException e) { e.printStackTrace(); } sseEmitterMap.put(uid, sseEmitter); log.info("[{}]創(chuàng)建sse連接成功!", uid); return sseEmitter; } /** * 給指定用戶發(fā)送消息 * */ public boolean sendMessage(String uid,String messageId, String message) { if (StrUtil.isBlank(message)) { log.info("參數(shù)異常,msg為null", uid); return false; } SseEmitter sseEmitter = sseEmitterMap.get(uid); if (sseEmitter == null) { log.info("消息推送失敗uid:[{}],沒有創(chuàng)建連接,請重試。", uid); return false; } try { sseEmitter.send(SseEmitter.event().id(messageId).reconnectTime(1*60*1000L).data(message)); log.info("用戶{},消息id:{},推送成功:{}", uid,messageId, message); return true; }catch (Exception e) { sseEmitterMap.remove(uid); log.info("用戶{},消息id:{},推送異常:{}", uid,messageId, e.getMessage()); sseEmitter.complete(); return false; } } /** * 斷開 * @param uid */ public void closeSse(String uid){ if (sseEmitterMap.containsKey(uid)) { SseEmitter sseEmitter = sseEmitterMap.get(uid); sseEmitter.complete(); sseEmitterMap.remove(uid); }else { log.info("用戶{} 連接已關(guān)閉",uid); } } }
- 創(chuàng)建SSE 端點:創(chuàng)建一個SseEmitter,用uid進行標識,uid可以是用戶標識符,也可以是業(yè)務標識符??梢岳斫鉃橥ㄐ判诺罉俗R。
- 通過端點發(fā)送事件:可以定時或在事件發(fā)生時調(diào)用sseEmitter.send()方法來發(fā)送事件。
- 關(guān)閉端點連接
四、編寫Controller
@Controller public class IndexAction { @Autowired private SseClient sseClient; @GetMapping("/") public String index(ModelMap model) { String uid = IdUtil.fastUUID(); model.put("uid",uid); return "index"; } @CrossOrigin @GetMapping("/createSse") public SseEmitter createConnect(String uid) { return sseClient.createSse(uid); } @CrossOrigin @GetMapping("/sendMsg") @ResponseBody public String sseChat(String uid) { for (int i = 0; i < 10; i++) { sseClient.sendMessage(uid, "no"+i,IdUtil.fastUUID()); } return "ok"; } /** * 關(guān)閉連接 */ @CrossOrigin @GetMapping("/closeSse") public void closeConnect(String uid ){ sseClient.closeSse(uid); } }
1,打開頁面默認頁面,傳遞端點標識。
2,連接端點(/createSse),頁面需要使用
3,通過ajax(/sendMsg),觸發(fā)后端業(yè)務(循環(huán)十條數(shù)據(jù)發(fā)往頁面),向頁面發(fā)送消息。
4,主動關(guān)閉連接(/closeSse)
五、前端接收與處理
HTML & JavaScript
在前端頁面,使用EventSource API訂閱SSE endpoint:
Html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <div id="con"></div> <script> let chat = document.getElementById("con"); if (window.EventSource) { //創(chuàng)建sse eventSource = new EventSource(`/createSse?uid=${uid}`); eventSource.onopen = function (event) { console.log('SSE鏈接成功'); } eventSource.onmessage = function (event) { if(event.data){ chat.innerHTML += event.data + '<br/>'; //console.log('后端返回的數(shù)據(jù):', data.value); } } eventSource.onerror = (error) => { console.log('SSE鏈接失敗'); }; } else { alert("你的瀏覽器不支持SSE"); } </script> </body> </html>
在這個例子中,前端每接收到一次SSE推送的事件,就會在id為"con"的元素中追加數(shù)據(jù)。
六、注意事項
- 當客戶端斷開連接時,SseEmitter會拋出IOException,所以務必捕獲并處理這種異常,通常情況下我們會調(diào)用
emitter.complete()
或emitter.completeWithError()
來關(guān)閉SseEmitter。 - SSE連接是持久性的,長時間保持連接可能需要處理超時和重連問題。
- 考慮到資源消耗,對于大量的并發(fā)客戶端,可能需要采用連接池或者其他優(yōu)化策略。
總結(jié),Spring Boot中利用SSE實現(xiàn)實時數(shù)據(jù)推送既簡單又實用,特別適合實時更新頻率不高、實時性要求不嚴苛的場景。同時,在高并發(fā)場景下需要注意資源管理和優(yōu)化策略的選擇。
到此這篇關(guān)于Spring Boot中使用Server-Sent Events (SSE) 實現(xiàn)實時數(shù)據(jù)推送教程的文章就介紹到這了,更多相關(guān)SpringBoot 實時數(shù)據(jù)推送內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MyBatis?超詳細講解動態(tài)SQL的實現(xiàn)
動態(tài)?SQL?是?MyBatis?的強大特性之一。如果你使用過?JDBC?或其它類似的框架,你應該能理解根據(jù)不同條件拼接?SQL?語句有多痛苦,例如拼接時要確保不能忘記添加必要的空格,還要注意去掉列表最后一個列名的逗號。利用動態(tài)?SQL,可以徹底擺脫這種痛苦2022-03-03詳解Spring Boot對 Apache Pulsar的支持
Spring Boot通過提供spring-pulsar和spring-pulsar-reactive自動配置支持Apache Pulsar,類路徑中這些依賴存在時,Spring Boot自動配置命令式和反應式Pulsar組件,PulsarClient自動注冊,默認連接本地Pulsar實例,感興趣的朋友一起看看吧2024-11-11Spring Boot中使用Actuator的/info端點輸出Git版本信息
這篇文章主要介紹了Spring Boot中使用Actuator的/info端點輸出Git版本信息,需要的朋友可以參考下2017-06-06