使用SSE流式輸出實(shí)戰(zhàn)記錄(Javaweb前后端)
一.什么是SSE?
SSE(Server-Sent Events)是一種在Web應(yīng)用中實(shí)現(xiàn)單向?qū)崟r通信的技術(shù)。通過SSE,服務(wù)器可以主動向客戶端發(fā)送更新,而無需客戶端不斷發(fā)起請求。這對于需要實(shí)時更新的應(yīng)用場景(如聊天應(yīng)用、實(shí)時數(shù)據(jù)監(jiān)控等)非常有用。(服務(wù)端單向傳輸客戶端,長連接)
主要特點(diǎn):
- 單向通信:數(shù)據(jù)從服務(wù)器流向客戶端,客戶端不能直接向服務(wù)器發(fā)送數(shù)據(jù)。
- 基于HTTP:SSE使用HTTP協(xié)議,簡單易用,支持瀏覽器。
- 自動重連:如果連接丟失,瀏覽器會自動嘗試重連。
- 文本格式:數(shù)據(jù)以文本流的形式發(fā)送,通常是UTF-8編碼。
使用場景:頻繁的更新數(shù)據(jù),低延遲,單向通信
二.SSE的實(shí)現(xiàn)過程:

首先前端給后端發(fā)送請求,隨后后端會返回響應(yīng)文本流創(chuàng)建連接,然后后端就可以持續(xù)的長連接進(jìn)行給前端傳輸數(shù)據(jù)。
SSE 基于 HTTP 協(xié)議,利用了其長連接特性,通過瀏覽器向服務(wù)器發(fā)送一個 HTTP 請求,建立一條持久化的連接。SSE 可以傳輸文本和二進(jìn)制格式的數(shù)據(jù),但只支持單向數(shù)據(jù)流,即只能由服務(wù)器向客戶端推送數(shù)據(jù)。
三.SSE的前端實(shí)現(xiàn):
1.創(chuàng)建 EventSource 對象:
通過指定服務(wù)器端的SSE URL,創(chuàng)建一個
EventSource實(shí)例。
const eventSource = new EventSource('http://localhost:8080/sse/chat');2.處理接收到的信息:
使用
onmessage事件處理程序接收服務(wù)器發(fā)送的消息。
eventSource.onmessage = function(event) {
console.log('Received:', event.data);
// 在這里處理接收到的數(shù)據(jù),比如更新UI
};
接收到的event是一個MessageEvent對象,它包含以下重要屬性:
data:這是服務(wù)器發(fā)送的消息內(nèi)容,通常是文本或JSON字符串。origin:表示事件來源的URI,通常是服務(wù)器的URL。lastEventId:如果瀏覽器重新連接,它會包含最后接收到的事件ID,這在處理丟失消息時非常有用。
eventSource.onmessage = function(event) {
// event.data的結(jié)果假設(shè)是 '{"name": "Alice", "age": 25}'
console.log('Received:', event.data);
// 如果是JSON格式,解析后 message 是一個對象: { name: "Alice", age: 25 }
const message = JSON.parse(event.data);
// 在這里更新UI或進(jìn)行其他操作
};
SSE發(fā)送的數(shù)據(jù)是以文本流的形式傳輸?shù)模ǔJ亲址ɡ?,JSON格式的字符串)。如果服務(wù)器發(fā)送的是JSON格式的字符串(例如 {"key": "value"}),那么你可以使用 JSON.parse() 將這個字符串轉(zhuǎn)換為JavaScript對象,以便更方便地訪問數(shù)據(jù)。
3.處理特定事件:
如果服務(wù)器發(fā)送了不同類型的事件,可以使用
oneventname處理特定事件。
eventSource.addEventListener('customEvent', function(event) {
console.log('Custom event received:', event.data);
});
4.處理連接錯誤問題:
可以使用
onerror事件處理程序來處理連接錯誤。
eventSource.onerror = function(event) {
console.error('EventSource failed:', event);
// 可以在這里實(shí)現(xiàn)重連邏輯
};
5.關(guān)閉連接:
如果不再需要接收消息,可以通過調(diào)用
close方法來關(guān)閉連接。
eventSource.close();
四.SSE的后端實(shí)現(xiàn):
1.導(dǎo)入web依賴:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
2.創(chuàng)建Controller類:
package com.itheima.bigdealboot.controller;
import lombok.extern.slf4j.Slf4j;
import org.springframework.http.MediaType;
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.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
@RestController
@RequestMapping("/sse")
@Slf4j
public class SSEController {
@GetMapping(value = "/chat", produces = MediaType.TEXT_EVENT_STREAM_VALUE)
public SseEmitter sseChat(){
SseEmitter emitter = new SseEmitter();
// 創(chuàng)建一個定時任務(wù)
ScheduledExecutorService executor = Executors.newSingleThreadScheduledExecutor();
executor.scheduleAtFixedRate(() -> {
try {
// 發(fā)送當(dāng)前時間戳
emitter.send(System.currentTimeMillis());
} catch (IOException e) {
// 處理異常(如客戶端關(guān)閉連接)
emitter.completeWithError(e);
}
}, 0, 1, TimeUnit.SECONDS); // 每秒發(fā)送一次
// 處理連接關(guān)閉
emitter.onCompletion(() -> executor.shutdown());
emitter.onTimeout(() -> {
emitter.complete();
executor.shutdown();
});
return emitter; // 返回SseEmitter實(shí)例
}
}
- SseEmitter實(shí)例:創(chuàng)建
SseEmitter對象,準(zhǔn)備向前端發(fā)送事件。 - 定時任務(wù):使用
ScheduledExecutorService定期發(fā)送數(shù)據(jù)(如當(dāng)前時間戳)。如果發(fā)送失?。ɡ?,客戶端關(guān)閉連接),則調(diào)用completeWithError()處理異常。 - 連接關(guān)閉處理:
onCompletion():當(dāng)連接正常關(guān)閉時調(diào)用,執(zhí)行資源清理。onTimeout():在連接超時時關(guān)閉連接。
3.SeeEmitter的介紹以及使用方法:
SseEmitter是 Spring框架提供的一個類,用于支持 Server-Sent Events (SSE)。
(1). 構(gòu)造方法
SseEmitter():創(chuàng)建一個新的SseEmitter實(shí)例,使用默認(rèn)的超時值。SseEmitter(Long timeout):創(chuàng)建一個新的SseEmitter實(shí)例,設(shè)置指定的超時時間(毫秒)。
(2). 發(fā)送數(shù)據(jù)
send(Object data):發(fā)送數(shù)據(jù)到客戶端。send(Object data, MediaType mediaType):發(fā)送數(shù)據(jù)到客戶端,并指定數(shù)據(jù)的媒體類型。send(SseEvent event):發(fā)送一個SseEvent對象到客戶端。
(3). 關(guān)閉連接
complete():正常完成事件流,關(guān)閉連接。completeWithError(Throwable throwable):由于錯誤完成事件流,并關(guān)閉連接。completeWithError(String message):由于錯誤完成事件流,并關(guān)閉連接,提供錯誤信息。
(4). 連接狀態(tài)處理
onCompletion(Runnable callback):注冊連接完成的回調(diào)函數(shù)。onTimeout(Runnable callback):注冊連接超時的回調(diào)函數(shù)。
(5). 獲取超時時間
getTimeout():返回當(dāng)前的超時時間(毫秒)。
(6). 其他
isCompleted():檢查SseEmitter是否已完成。isExpired():檢查SseEmitter是否已過期。
總結(jié)
到此這篇關(guān)于使用SSE流式輸出(Javaweb前后端)的文章就介紹到這了,更多相關(guān)Javaweb前后端SSE流式輸出內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring創(chuàng)建BeanDefinition之路徑掃描詳解
這篇文章主要介紹了Spring創(chuàng)建BeanDefinition之路徑掃描方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2025-04-04
idea2019導(dǎo)入maven項(xiàng)目中的某些問題及解決方法
這篇文章主要介紹了idea2019導(dǎo)入maven項(xiàng)目中的某些問題,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-08-08
Spring AI 實(shí)現(xiàn) STDIO和SSE MCP Server的過
STDIO方式是基于進(jìn)程間通信,MCP Client和MCP Server運(yùn)行在同一主機(jī),主要用于本地集成、命令行工具等場景,這篇文章主要介紹了Spring AI 實(shí)現(xiàn) STDIO和SSE MCP Server,需要的朋友可以參考下2025-05-05
SpringBoot處理form-data表單接收對象數(shù)組的方法
form-data則是一種更加靈活的編碼方式,它可以處理二進(jìn)制數(shù)據(jù)(如圖片、文件等)以及文本數(shù)據(jù),這篇文章主要介紹了SpringBoot處理form-data表單接收對象數(shù)組,需要的朋友可以參考下2023-11-11
Spring向頁面?zhèn)髦岛徒邮茼撁鎮(zhèn)鬟^來的參數(shù)詳解
這篇文章主要給大家介紹了關(guān)于Spring向頁面?zhèn)髦岛徒邮茼撁鎮(zhèn)鬟^來的參數(shù)的相關(guān)資料,文中介紹的非常詳細(xì),對大家具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起看看吧。2017-06-06

