Java中ResponseBodyEmitter的實(shí)現(xiàn)
前言
在開(kāi)發(fā)高并發(fā)應(yīng)用或處理長(zhǎng)時(shí)間任務(wù)時(shí),服務(wù)端需要向客戶(hù)端實(shí)時(shí)推送數(shù)據(jù),而不是一次性將所有結(jié)果返回。Spring 提供了一種優(yōu)雅的解決方案:ResponseBodyEmitter
。它適用于需要逐步發(fā)送響應(yīng)數(shù)據(jù)的場(chǎng)景,比如進(jìn)度條更新、實(shí)時(shí)日志輸出、消息流等。本文將深入講解 ResponseBodyEmitter
的核心概念、使用場(chǎng)景、完整示例以及注意事項(xiàng),幫助初學(xué)者快速掌握其使用方法。
什么是 ResponseBodyEmitter?
ResponseBodyEmitter
是 Spring MVC 提供的一個(gè)類(lèi),用于實(shí)現(xiàn)服務(wù)端向客戶(hù)端分塊推送響應(yīng)數(shù)據(jù)。它是異步非阻塞的,可以在響應(yīng)未完成時(shí)多次向客戶(hù)端發(fā)送部分?jǐn)?shù)據(jù),而無(wú)需等待任務(wù)完成。
特性
- 異步非阻塞:支持異步任務(wù),可以有效提高服務(wù)端吞吐量。
- 實(shí)時(shí)性:能夠逐步將數(shù)據(jù)推送給客戶(hù)端,適用于實(shí)時(shí)數(shù)據(jù)更新場(chǎng)景。
- 兼容性:基于標(biāo)準(zhǔn)的 HTTP 協(xié)議,客戶(hù)端無(wú)需特殊支持。
ResponseBodyEmitter 的基本用法
核心方法
send(Object data)
:向客戶(hù)端發(fā)送數(shù)據(jù),可以多次調(diào)用。complete()
:結(jié)束響應(yīng)流,表示數(shù)據(jù)發(fā)送完畢。onTimeout(Runnable callback)
:設(shè)置超時(shí)回調(diào)函數(shù)。onCompletion(Runnable callback)
:設(shè)置完成回調(diào)函數(shù)。
典型使用場(chǎng)景
- 實(shí)時(shí)日志輸出:將長(zhǎng)時(shí)間運(yùn)行任務(wù)的日志實(shí)時(shí)返回給客戶(hù)端。
- 進(jìn)度條更新:在任務(wù)執(zhí)行過(guò)程中動(dòng)態(tài)更新任務(wù)進(jìn)度。
- 數(shù)據(jù)流式加載:用于大數(shù)據(jù)分片加載,比如分頁(yè)查詢(xún)實(shí)時(shí)返回結(jié)果。
實(shí)戰(zhàn):實(shí)現(xiàn)一個(gè)實(shí)時(shí)推送的示例
下面我們通過(guò)一個(gè)完整的例子,演示如何使用 ResponseBodyEmitter
實(shí)現(xiàn)任務(wù)進(jìn)度實(shí)時(shí)推送功能。
示例代碼
1. 創(chuàng)建 Controller
@RestController @RequestMapping("/api/progress") public class ProgressController { @GetMapping("/start") public ResponseBodyEmitter startTask() { // 創(chuàng)建一個(gè) ResponseBodyEmitter 實(shí)例 ResponseBodyEmitter emitter = new ResponseBodyEmitter(); // 模擬一個(gè)耗時(shí)任務(wù) new Thread(() -> { try { for (int i = 1; i <= 100; i += 10) { // 向客戶(hù)端發(fā)送進(jìn)度 emitter.send("Progress: " + i + "%\n"); Thread.sleep(1000); // 模擬任務(wù)耗時(shí) } emitter.complete(); // 任務(wù)完成,關(guān)閉連接 } catch (Exception e) { emitter.completeWithError(e); // 出現(xiàn)異常時(shí)通知客戶(hù)端 } }).start(); return emitter; // 返回 Emitter } }
2. 測(cè)試接口
可以使用 Postman、瀏覽器或客戶(hù)端代碼調(diào)用接口:
URL: http://localhost:8080/api/progress/start
客戶(hù)端會(huì)逐步接收到如下響應(yīng):
Progress: 10% Progress: 20% Progress: 30% ... Progress: 100%
深入分析
ResponseBodyEmitter 工作原理
- 服務(wù)端異步生成響應(yīng)數(shù)據(jù):任務(wù)執(zhí)行時(shí),調(diào)用
send()
方法將數(shù)據(jù)推送至客戶(hù)端。 - 分塊傳輸:數(shù)據(jù)以 HTTP 的**分塊編碼(Chunked Encoding)**方式傳輸,不會(huì)提前設(shè)置
Content-Length
,而是分段發(fā)送數(shù)據(jù)塊。 - 連接生命周期:通過(guò)
complete()
或completeWithError()
控制連接的關(guān)閉。
重要注意事項(xiàng)
- 支持的客戶(hù)端:大多數(shù)瀏覽器和 HTTP 客戶(hù)端庫(kù)支持分塊傳輸,但某些老舊的客戶(hù)端可能不支持。
- 超時(shí)設(shè)置:為了避免長(zhǎng)連接占用資源,可以為
ResponseBodyEmitter
設(shè)置超時(shí)時(shí)間:emitter.onTimeout(() -> emitter.complete());
- 線(xiàn)程安全:
ResponseBodyEmitter
的send()
方法是線(xiàn)程安全的,但需要注意控制任務(wù)線(xiàn)程的生命周期。 - 連接關(guān)閉:需要確保任務(wù)結(jié)束時(shí)調(diào)用
complete()
或completeWithError()
,否則可能導(dǎo)致資源泄露。
擴(kuò)展:與 Streaming 和 SSE 的對(duì)比
- Streaming:直接通過(guò)
OutputStream
向客戶(hù)端寫(xiě)入數(shù)據(jù),靈活性高,但需手動(dòng)處理流的關(guān)閉。 - Server-Sent Events (SSE):基于
text/event-stream
,適用于服務(wù)端事件推送,客戶(hù)端需支持 SSE。 - ResponseBodyEmitter:更通用,適用于任何支持 HTTP 的客戶(hù)端,且易于與 Spring 集成。
總結(jié)
ResponseBodyEmitter
是 Spring 提供的輕量級(jí)流式傳輸解決方案,能有效提升高并發(fā)和實(shí)時(shí)性場(chǎng)景的用戶(hù)體驗(yàn)。通過(guò)本文的講解和示例,相信大家已經(jīng)掌握了它的使用技巧和注意事項(xiàng)。在實(shí)際項(xiàng)目中,不妨嘗試將其應(yīng)用于實(shí)時(shí)日志、進(jìn)度更新等場(chǎng)景,讓你的應(yīng)用更加智能、高效。
到此這篇關(guān)于Java中ResponseBodyEmitter的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Java ResponseBodyEmitter內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringCloud使用Zookeeper作為注冊(cè)中心
這篇文章主要介紹了SpringCloud如何使用Zookeeper作為注冊(cè)中心,幫助大家更好的理解和學(xué)習(xí)使用Zookeeper,感興趣的朋友可以了解下2021-04-04nacos客戶(hù)端一致性hash負(fù)載需求實(shí)現(xiàn)
這篇文章主要介紹了nacos客戶(hù)端一致性hash負(fù)載的需求實(shí)現(xiàn)過(guò)程及步驟詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2022-02-02java開(kāi)發(fā)的工廠方法模式及抽象工廠驗(yàn)證示例
這篇文章主要為大家介紹了java開(kāi)發(fā)中的工廠方法模式以及抽象工廠的驗(yàn)證示例,有需要的朋友可以借鑒參考下希望能夠有所幫助祝大家多多進(jìn)步2021-10-10Java中的WeakHashMap概念原理以及簡(jiǎn)單案例
這篇文章主要介紹了Java中的WeakHashMap概念原理以及簡(jiǎn)單案例,WeakHashMap使用了軟引用結(jié)構(gòu),它的對(duì)象在垃圾回收時(shí)會(huì)被刪除,垃圾回收是優(yōu)先級(jí)非常低的線(xiàn)程,不能被顯示調(diào)用,當(dāng)內(nèi)存不足的時(shí)候會(huì)啟用,需要的朋友可以參考下2023-09-09Maven分模塊開(kāi)發(fā)執(zhí)行指令失敗的問(wèn)題
Maven分模塊開(kāi)發(fā),行指令失敗,modules.module[3]‘ specifies duplicate child module maven_dao @ line 29, column 1的問(wèn)題,本文給大家分享解決方法,感興趣的朋友跟隨小編一起看看吧2020-09-09