如何使用 SseEmitter 實(shí)現(xiàn) Spring Boot 后端的流式傳輸和前端的數(shù)據(jù)接收
使用 SseEmitter 實(shí)現(xiàn) Spring Boot 后端的流式傳輸和前端的數(shù)據(jù)接收
在構(gòu)建現(xiàn)代 Web 應(yīng)用時(shí),實(shí)時(shí)數(shù)據(jù)推送是一個(gè)非常重要的需求,比如 AI 對(duì)話流式響應(yīng)、實(shí)時(shí)通知推送等。而在 Spring Boot 中,SseEmitter 是一個(gè)非常方便的工具,它可以讓后端向前端進(jìn)行流式數(shù)據(jù)傳輸(Server-Sent Events,簡稱 SSE)。本文將介紹如何使用 SseEmitter 創(chuàng)建流式傳輸,并實(shí)現(xiàn)后端與前端的高效交互。
1. 什么是 SseEmitter?
SseEmitter 是 Spring Web 提供的一個(gè)用于服務(wù)器推送事件(SSE, Server-Sent Events)的工具。它可以讓服務(wù)器端不斷向客戶端推送數(shù)據(jù),而不需要客戶端不斷輪詢。
相比于 WebSocket,SSE 具有以下優(yōu)點(diǎn):
- 基于 HTTP 連接,無需額外的協(xié)議支持,前端可以直接使用
EventSource進(jìn)行接收。 - 支持自動(dòng)重連,如果連接斷開,瀏覽器會(huì)自動(dòng)嘗試重新建立連接。
- 更適合單向推送數(shù)據(jù)的場景,如 AI 生成流式文本、服務(wù)器消息推送等。
2. 在 Spring Boot 中使用 SseEmitter
2.1 添加依賴
如果你的項(xiàng)目使用的是 Spring MVC,無需額外的依賴。但如果你使用的是 Spring WebFlux,請(qǐng)確保你的 pom.xml 中包含以下依賴(僅 WebFlux 需要):
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-webflux</artifactId>
</dependency>2.2 創(chuàng)建后端流式接口
java的代碼示例如下:
@RestController
@RequestMapping("/api/chat")
public class ChatController {
@GetMapping("/stream")
public SseEmitter streamResponse() {
SseEmitter emitter = new SseEmitter(0L); // 0L 表示永不超時(shí)
Executors.newSingleThreadExecutor().execute(() -> {
try {
for (int i = 1; i <= 5; i++) {
emitter.send("消息 " + i);
Thread.sleep(1000); // 模擬延遲
}
emitter.complete();
} catch (Exception e) {
emitter.completeWithError(e);
}
});
return emitter;
}
}3. 前端如何接收流式數(shù)據(jù)
?? 使用 fetch 處理 SSE(流式數(shù)據(jù)),fetch 默認(rèn)不會(huì)處理流式數(shù)據(jù),因此我們需要手動(dòng)解析 ReadableStream 以逐步接收數(shù)據(jù)。
? 前端代碼(基于 fetch/infetch)
async function fetchStreamData() {
const response = await fetch("/api/chat/stream");
// 確保服務(wù)器支持流式數(shù)據(jù)
if (!response.ok) {
throw new Error(`HTTP 錯(cuò)誤!狀態(tài)碼: ${response.status}`);
}
const reader = response.body.getReader();
const decoder = new TextDecoder("utf-8");
// 讀取流式數(shù)據(jù)
while (true) {
const { done, value } = await reader.read();
if (done) break;
// 解碼并輸出數(shù)據(jù)
const text = decoder.decode(value, { stream: true });
console.log("收到數(shù)據(jù):", text);
}
console.log("流式傳輸完成");
}
// 調(diào)用流式請(qǐng)求
fetchStreamData().catch(console.error);?? 為什么使用 fetch + ReadableStream?
- 兼容性更好:fetch API 是現(xiàn)代瀏覽器的標(biāo)準(zhǔn)方法,相比 EventSource 可處理更多類型的流式數(shù)據(jù)(如 JSON、HTML、二進(jìn)制)。
- 更靈活:可以手動(dòng)控制數(shù)據(jù)讀取,支持自定義解析方式。
- 適用于 AI 大模型返回:如果你的后端是 ChatGPT/DeepSeek API 這樣的 AI 接口,fetch + stream 是最佳方式。
到此這篇關(guān)于使用 SseEmitter 實(shí)現(xiàn) Spring Boot 后端的流式傳輸和前端的數(shù)據(jù)接收的文章就介紹到這了,更多相關(guān)Spring Boot SseEmitter流式傳輸和數(shù)據(jù)接收內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java 數(shù)組內(nèi)置函數(shù)toArray詳解
這篇文章主要介紹了Java 數(shù)組內(nèi)置函數(shù)toArray詳解,文本詳細(xì)的講解了toArray底層的代碼和文檔,需要的朋友可以參考下2021-06-06
Spring+SpringMVC+MyBatis深入學(xué)習(xí)及搭建(二)之MyBatis原始Dao開發(fā)和mapper代理開發(fā)
這篇文章主要介紹了Spring+SpringMVC+MyBatis深入學(xué)習(xí)及搭建(二)之MyBatis原始Dao開發(fā)和mapper代理開發(fā),需要的朋友可以參考下2017-05-05
Java并發(fā)Futures和Callables類實(shí)例詳解
Callable對(duì)象返回Future對(duì)象,該對(duì)象提供監(jiān)視線程執(zhí)行的任務(wù)進(jìn)度的方法, Future對(duì)象可用于檢查Callable的狀態(tài),然后線程完成后從Callable中檢索結(jié)果,這篇文章給大家介紹Java并發(fā)Futures和Callables類的相關(guān)知識(shí),感興趣的朋友一起看看吧2024-05-05
spring學(xué)習(xí)之util:properties的使用
這篇文章主要介紹了spring學(xué)習(xí)之util:properties的使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01
在java中由類名和方法名字符串實(shí)現(xiàn)其調(diào)用方式
這篇文章主要介紹了在java中由類名和方法名字符串實(shí)現(xiàn)其調(diào)用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-09-09

