Springboot快速集成sse服務(wù)端推流(最新整理)
1、前言
如果項目中有一個場景,假設(shè)對接ChatGPT或?qū)犹鞖忸惤涌诘臅r候,需要服務(wù)端主動往客戶端進(jìn)行消息推送或推流。通常的做法有:
- 客戶端提供接收數(shù)據(jù)接口,服務(wù)端開啟定時輪詢,定時向客戶端發(fā)起http請求
- 客戶端提供定時輪詢服務(wù),定時向服務(wù)端發(fā)起http請求接口
- 使用websocket實時通訊
那么今天再介紹另一種機制:SSE,也就是服務(wù)器發(fā)送事件機制。
2、什么是SSE
SSE(Server-Sent Events)是一種允許服務(wù)器向客戶端推送實時數(shù)據(jù)的技術(shù),它建立在 HTTP 和簡單文本格式之上,提供了一種輕量級的服務(wù)器推送方式,通常也被稱為“事件流”(Event Stream)。他通過在客戶端和服務(wù)端之間建立一個長連接,并通過這條連接實現(xiàn)服務(wù)端和客戶端的消息實時推送。
2.1、技術(shù)原理
SSE是建立在HTTP協(xié)議之上的,所以原理比較簡單,也與HTTP原理類似:
1)建立連接:
客戶端通過普通的 HTTP 請求向服務(wù)器發(fā)起連接請求,類似于普通的 Web 請求。這個請求的關(guān)鍵在于使用了 text/event-stream 的 MIME 類型,告知服務(wù)器該請求是 SSE 請求。
httpCopy codeGET /sse/stream HTTP/1.1 Host: example.com Accept: text/event-stream
2)服務(wù)器處理請求:
服務(wù)器接收到 SSE 請求后,會在連接上保持打開狀態(tài),不會立即關(guān)閉。這是與普通的請求-響應(yīng)模式的主要不同之處。服務(wù)器端通過這個持久連接向客戶端發(fā)送數(shù)據(jù)。
3)數(shù)據(jù)推送:
服務(wù)器端通過打開的連接,周期性地向客戶端發(fā)送消息。這些消息以文本的形式發(fā)送,并遵循一定的格式,通常以 data 字段表示消息內(nèi)容。
httpCopy codeHTTP/1.1 200 OK Content-Type: text/event-stream data: This is a message\n\n
上述例子中,data 字段包含了實際的消息內(nèi)容,兩個換行符(\n\n)表示消息的結(jié)束。
4)客戶端接收消息:
客戶端通過監(jiān)聽連接的 message 事件來接收服務(wù)器推送的消息。一旦接收到消息,客戶端可以采取相應(yīng)的操作,例如更新界面內(nèi)容。
javascriptCopy codeconst eventSource = new EventSource('/sse/stream'); eventSource.onmessage = function (event) { console.log('Received message:', event.data); // 處理消息,例如更新界面 };
5)連接關(guān)閉:
當(dāng)服務(wù)器端不再需要向客戶端推送消息時,或者發(fā)生錯誤時,服務(wù)器可以關(guān)閉連接??蛻舳艘部梢酝ㄟ^調(diào)用 eventSource.close() 來關(guān)閉連接。
2.2、SSE和WebSocket
提到SSE,那自然要提一下WebSocket了。WebSocket是一種HTML5提供的全雙工通信協(xié)議(指可以在同一時間內(nèi)允許兩個設(shè)備之間進(jìn)行雙向發(fā)送和接收數(shù)據(jù)的通信協(xié)議),基于TCP協(xié)議,并復(fù)用HTTP的握手通道(允許一次TCP連接中傳輸多個HTTP請求和相應(yīng)),常用于瀏覽器與服務(wù)器之間的實時通信。
SSE和WebSocket盡管功能類似,都是用來實現(xiàn)服務(wù)器向客戶端實時推送數(shù)據(jù)的技術(shù),但還是有一定區(qū)別:
2.2.1、SSE (Server-Sent Events)
- 簡單性:SSE 使用簡單的 HTTP 協(xié)議,通常建立在標(biāo)準(zhǔn)的 HTTP 或 HTTPS 連接之上。這使得它對于一些簡單的實時通知場景非常適用,特別是對于服務(wù)器向客戶端單向推送數(shù)據(jù)。
- 兼容性:SSE 在瀏覽器端具有較好的兼容性,因為它是基于標(biāo)準(zhǔn)的 HTTP 協(xié)議的。即使在一些不支持 WebSocket 的環(huán)境中,SSE 仍然可以被支持。
- 適用范圍:SSE 適用于服務(wù)器向客戶端單向推送通知,例如實時更新、事件通知等。但它僅支持從服務(wù)器到客戶端的單向通信,客戶端無法直接向服務(wù)器發(fā)送消息。
2.2.2、WebSocket
全雙工通信: WebSocket 提供了全雙工通信,允許客戶端和服務(wù)器之間進(jìn)行雙向?qū)崟r通信。這使得它適用于一些需要雙向數(shù)據(jù)交換的應(yīng)用,比如在線聊天、實時協(xié)作等。
低延遲:WebSocket 的通信開銷相對較小,因為它使用單一的持久連接,而不像 SSE 需要不斷地創(chuàng)建新的連接。這可以降低通信的延遲。
適用范圍: WebSocket 適用于需要實時雙向通信的應(yīng)用,特別是對于那些需要低延遲、高頻率消息交換的場景。
2.2.3、選擇 SSE 還是 WebSocket?
簡單通知場景:如果你只需要服務(wù)器向客戶端推送簡單的通知、事件更新等,而不需要客戶端與服務(wù)器進(jìn)行雙向通信,那么 SSE 是一個簡單而有效的選擇。
雙向通信場景:如果你的應(yīng)用需要實現(xiàn)實時雙向通信,例如在線聊天、協(xié)作編輯等,那么 WebSocket 是更合適的選擇。
兼容性考慮: 如果你的應(yīng)用可能在一些不支持 WebSocket 的環(huán)境中運行,或者需要考慮到更廣泛的瀏覽器兼容性,那么 SSE 可能是一個更可行的選擇。
3、Springboot快速集成
3.1、添加依賴
Springboot項目中,sse不需要額外添加依賴,引用了web相關(guān)的springboot依賴即可:
<dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> </dependencies>
3.2、創(chuàng)建SSE控制器
這里簡單創(chuàng)建一個控制器類,用于處理SSE請求。在JAVA中通常使用SSEmitter來實現(xiàn)sse的消息推送。
package com.example.springbootsse.controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; import org.springframework.web.servlet.mvc.method.annotation.SseEmitter; import java.io.IOException; import java.util.Date; @RestController @RequestMapping("/sse") public class SSEmitterController { @GetMapping("/stream") public SseEmitter stream() { // 用于創(chuàng)建一個 SSE 連接對象 SseEmitter emitter = new SseEmitter(); // 在后臺線程中模擬實時數(shù)據(jù) new Thread(() -> { try { for (int i = 0; i < 10; i++) { // emitter.send() 方法向客戶端發(fā)送消息 // 使用SseEmitter.event()創(chuàng)建一個事件對象,設(shè)置事件名稱和數(shù)據(jù) emitter.send(SseEmitter.event().name("message").data("[" + new Date() + "] Data #" + i)); Thread.sleep(1000); } // 數(shù)據(jù)發(fā)送完成后,關(guān)閉連接 emitter.complete(); } catch (IOException | InterruptedException e) { // 發(fā)生錯誤時,關(guān)閉連接并報錯 emitter.completeWithError(e); } }).start(); return emitter; } }
查看執(zhí)行結(jié)果,可以看到每一秒服務(wù)端都會自動像客戶端推送messag消息:
我們來關(guān)注下SSEmitter這個類,SseEmitter 是 Spring Framework 中用于實現(xiàn) Server-Sent Events(SSE)的一個類。它允許服務(wù)器向客戶端推送數(shù)據(jù),通過建立一個持久連接,實現(xiàn)服務(wù)器向客戶端的實時單向通信。在 Spring 框架中,SseEmitter 類通常用于處理 SSE 請求,推送事件給客戶端。
3.2.1、SSEmitter創(chuàng)建實例
SSEmitter提供了兩個構(gòu)造函數(shù)用于創(chuàng)建實例。在創(chuàng)建實例時,我們可以指定超時時間timeout,如果傳0或使用無參構(gòu)造,則表示永不過期。連接超時是指在一段時間內(nèi)沒有數(shù)據(jù)傳輸時,連接將被認(rèn)為是超時的,并自動關(guān)閉。
3.2.2、SSEmitter API
除此以外,SSEmitter還提供了幾種API,如上面例子中使用到的:
- emitter.send() 方法向客戶端發(fā)送消息。
- SseEmitter.event() 創(chuàng)建一個事件對象,設(shè)置事件名稱和數(shù)據(jù)。
- emitter.complete() 表示數(shù)據(jù)發(fā)送完成后關(guān)閉連接。
- emitter.completeWithError(e) 在發(fā)生錯誤時關(guān)閉連接并報錯。
3.2.3、SSEmitter注冊回調(diào)
SseEmitter 可以通過注冊回調(diào)函數(shù)來處理服務(wù)器端發(fā)往客戶端的事件。當(dāng)服務(wù)器端有新的數(shù)據(jù)需要推送給客戶端時,注冊的回調(diào)函數(shù)將會被調(diào)用。SSEmitter繼承了ResponseBodyEmitter,提供的一系列注冊回調(diào)函數(shù)有:
示例代碼:
package com.example.springbootsse.controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; import org.springframework.web.servlet.mvc.method.annotation.SseEmitter; import java.io.IOException; import java.util.Date; @RestController @RequestMapping("/sse") public class SSEmitterController { @GetMapping("/stream") public SseEmitter stream() { // 3S超時 SseEmitter emitter = new SseEmitter(10000L); // 注冊回調(diào)函數(shù),處理服務(wù)器向客戶端推送的消息 emitter.onCompletion(() -> { System.out.println("Connection completed"); // 在連接完成時執(zhí)行一些清理工作 }); emitter.onTimeout(() -> { System.out.println("Connection timeout"); // 在連接超時時執(zhí)行一些處理 emitter.complete(); }); // 在后臺線程中模擬實時數(shù)據(jù) new Thread(() -> { try { for (int i = 0; i < 10; i++) { emitter.send(SseEmitter.event().name("message").data("[" + new Date() + "] Data #" + i)); Thread.sleep(1000); } emitter.complete(); // 數(shù)據(jù)發(fā)送完成后,關(guān)閉連接 } catch (IOException | InterruptedException e) { emitter.completeWithError(e); // 發(fā)生錯誤時,關(guān)閉連接并報錯 } }).start(); return emitter; } }
- onCompletion():在連接完成時候觸發(fā),可在連接完成時執(zhí)行一些清理工作
- onTimeout():當(dāng)連接超時時觸發(fā)
- onError():當(dāng)連接異常時觸發(fā)
- completeWithError(e):用于發(fā)生錯誤時,關(guān)閉連接并報錯
4、小結(jié)
其實SSE已經(jīng)出來很久了,但是熟知他的人卻很少,大多數(shù)項目中還是直接使用了websocket技術(shù)。直到最近ChatGPT火了之后,很多項目需要對接GPT進(jìn)行實時推流,才逐漸又被人提起。所以借此篇文章給自己掃盲一下。
到此這篇關(guān)于Springboot集成sse服務(wù)端推流的文章就介紹到這了,更多相關(guān)Springboot集成sse內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot服務(wù)正常啟動之后,訪問服務(wù)url無響應(yīng)問題及解決
這篇文章主要介紹了springboot服務(wù)正常啟動之后,訪問服務(wù)url無響應(yīng)問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-07-07SpringBoot中Mybatis注解一對多和多對多查詢實現(xiàn)示例
這篇文章主要介紹了SpringBoot中Mybatis注解一對多和多對多查詢的實現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03詳解Java中l(wèi)og4j.properties配置與加載應(yīng)用
這篇文章主要介紹了 log4j.properties配置與加載應(yīng)用的相關(guān)資料,需要的朋友可以參考下2018-02-02SpringBoot事務(wù)異步調(diào)用引發(fā)的bug解決
本文主要介紹了SpringBoot事務(wù)異步調(diào)用引發(fā)的bug解決,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06