Spring DeferredResult 實現長輪詢

1、背景

在項目開發中,有一個流程性的方法執行,這個方法會調用各種方法,可能會導致時間比較長 ,如果一直等待響應結果的話,可能會造成超時,如果直接使用異步的方式的話,前端無法知道整體流程什么時候會結束,

2、解決方案

使用了DeferredResult 的方式,設置超時時間,當流程執行完了沒有超過指定時間就可以直接返回結果,如果超過了指定時間就給前端先返回超時結果,并且指定一個唯一標志放到結果中返回,前端后續可以拿著這個唯一標志來輪詢,知道返回執行完成

關于 DeferredResult :請求的處理線程(即 tomcat 線程池的線程)不會等DeferredResult#setResult() 被調用才釋放,而是直接釋放了

也就是說 tomcat 線程安排好 DeferredResult 的一些配置后,不會等邏輯處理完(DeferredResult->setResult()的調用或者超時)。

而是直接釋放了,這樣 tomcat 線程就被回收到線程池中了,可以響應其他請求,不會傻傻地阻塞等著 DeferredResult->setResult() 被調用或超時。

我們都知道 tomcat 的線程池大小是有限的,如果我們的一些業務邏輯處理慢的話,會漸漸地占滿 tomcat 線程,這樣就無法處理新的請求,所以一些處理緩慢的業務我們會放到業務線程池中處理,但單純的放到業務線程池中處理的話,我們無法得知其什么時候處理完,也無法將處理完的結果和之前的請求匹配上,所以常做的方式就是輪詢。

而 DeferredResult 的做法就類似僅把事情安排好,不會管事情做好沒,tomcat 線程就釋放走了,注意此時不會給請求方(如瀏覽器)任何響應,而是將請求存放在一邊,等后面有結果了再把之前的請求拿來,把值響應給請求方。

用簡單的話來總結下 Spring DeferredResult :如果返回值類型是 DeferredResult 則表明其是異步請求,tomcat 線程不會等到應用程序處理完或者超時,而是會立即釋放線程。

而這個未處理完的請求則會暫存,tomcat 知曉其為異步請求,也不會對客戶端進行響應,直至 tomcat 線程掃描到請求超時或者應用線程將 result 塞入到 DeferredResult 中。

3、一些常用方法

public void onTimeout(Runnable callback)

public void onError(Consumer<Throwable> callback)

public void onCompletion(Runnable callback)

public boolean setResult(T result)

  • onTimeout():僅超時觸發。
  • onError():僅異步任務拋出異常時觸發。
  • onCompletion() 是兜底回調,無論何種結束方式都會執行,適合釋放共享資源(如移除緩存、關閉連接)
  • setResult() 設置返回結果集

4、具體簡單代碼實現

// -1 表示任務未完成 0 表示任務失敗 其它是具體值private static final Map<String,String> EXEC_CACHE  = new ConcurrentHashMap<>();@Overridepublic DeferredResult<Map<String, String>> exec() {DeferredResult<Map<String, String>> deferredResult = new DeferredResult<>(5000L);// 設置超時回調(如果任務未在 2 秒內完成,返回超時響應)String flag = UUID.randomUUID().toString();Map<String, String> result = new HashMap<>();deferredResult.onTimeout(() -> {result.put("code", "410");result.put("message", "任務未完成,已超時!");result.put("flag",flag);deferredResult.setResult(result);EXEC_CACHE.put(flag, "-1");});//執行具體任務CompletableFuture.runAsync(()->{try {//模擬執行耗時Thread.sleep(1000);//模擬獲到執行結果String execResult = UUID.randomUUID().toString();//放入緩存EXEC_CACHE.put(flag, execResult);//完成響應result.put("code", "200");result.put("message", "任務完成");result.put("flag",execResult);deferredResult.setResult(result);}catch (Exception e){//完成響應result.put("code", "500");result.put("message", "任務完成");result.put("flag",null);deferredResult.setResult(result);EXEC_CACHE.put(flag, "0");}});return deferredResult;}@Overridepublic String queryExecResult(String flag) {//前端根據具體值判斷要不要繼續輪詢return EXEC_CACHE.get(flag);}

執行超時情況

輪詢查詢

未超時返回

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/915892.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/915892.shtml
英文地址,請注明出處:http://en.pswp.cn/news/915892.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Python設計模式 - 橋接模式

定義 橋接模式是一種結構型設計模式&#xff0c;它的核心思想是將抽象部分與實現部分分離&#xff0c;使它們可以獨立變化。 結構抽象類&#xff08;Abstraction&#xff09;&#xff1a;定義抽象接口&#xff0c;持有實現部分的引用。具體抽象類&#xff08;Refined Abstracti…

【NLP輿情分析】基于python微博輿情分析可視化系統(flask+pandas+echarts) 視頻教程 - 用戶注冊實現

大家好&#xff0c;我是java1234_小鋒老師&#xff0c;最近寫了一套【NLP輿情分析】基于python微博輿情分析可視化系統(flaskpandasecharts)視頻教程&#xff0c;持續更新中&#xff0c;計劃月底更新完&#xff0c;感謝支持。今天講解用戶注冊實現 視頻在線地址&#xff1a; …

華為7月23日機考真題

&#x1f4cc; 點擊直達筆試專欄 &#x1f449;《大廠筆試突圍》 &#x1f4bb; 春秋招筆試突圍在線OJ 筆試突圍OJ](bishipass.com) 03. 山峰觀測站數據分析 問題描述 LYA是一名地理數據分析師&#xff0c;負責分析山峰觀測站收集的海拔高度數據。觀測站在一條直線上設置了…

圖像分析學習筆記(4):機器學習圖像特征與描述

圖像分析學習筆記&#xff08;4&#xff09;&#xff1a;機器學習圖像特征與描述深度學習基礎深度學習技巧深度模型構建深度學習基礎 深度學習概念&#xff1a;深度學習是機器學習的一個分支&#xff0c;它基于一系列算法&#xff0c;試圖通過使用多個處理層建立數據的高級抽象…

鎖付機器人,如何精準鎖附革新新能源鋰電裝配效率

其實呢&#xff0c;隨著科技的不斷發展&#xff0c;新能源電池、智能制造、精密裝配、工藝升級以及工業自動化這些領域都在飛速前進。新能源行業如今可是炙手可熱&#xff0c;中國新能源行業進入快速發展階段&#xff0c;就像一列高速行駛的火車&#xff0c;勢不可擋。在這個過…

Vue項目開發注意事項(包含node/npm/cnpm等)

事項一&#xff1a;項目代碼放在本地怎么運行起來 1、首先確定項目對應的node和npm版本 node下載地址 Index of /dist/https://nodejs.org/dist/ node 與 npm版本對應關系 Node.js — Node.js Releases 2、node卸載的時候&#xff0c;會自動把對應的npm卸載掉 情況1&…

GitHub:只支持 Git 作為唯一的版本庫格式進行托管

&#x1f90d; 前端開發工程師、技術日更博主、已過CET6 &#x1f368; 阿珊和她的貓_CSDN博客專家、23年度博客之星前端領域TOP1 &#x1f560; 牛客高級專題作者、打造專欄《前端面試必備》 、《2024面試高頻手撕題》、《前端求職突破計劃》 &#x1f35a; 藍橋云課簽約作者、…

秋招Day17 - Spring - MVC

Spring MVC有哪些核心組件&#xff1f;DispatcherServlet&#xff1a;前端控制器&#xff0c;所有HTTP請求首先經過它&#xff0c;分發請求到正確的處理器&#xff0c;并與其他組件協調。HandlerMapping&#xff1a;維護URL和處理器的映射關系Handler&#xff1a;處理器&#x…

使用mybatis實現模糊查詢和精準查詢切換的功能

1、首先在前端頁面添加勾選框&#xff08;name設置為check&#xff09;2、mybatis代碼當check勾選時&#xff0c;check不為null&#xff0c;走模糊查詢like當check未勾選時&#xff0c;check為null&#xff0c;走精準查詢 <if test"check ! null and check ! "&g…

Android模塊化實現方案深度分析

模塊化是現代 Android 開發應對項目復雜度激增、團隊協作效率、編譯速度瓶頸、功能復用與動態化等挑戰的核心架構思想。其核心目標是高內聚、低耦合、可插拔、易維護。 一、模塊化的核心價值與目標 降低復雜度&#xff1a; 將龐大單體應用拆分為獨立、職責清晰的模塊。加速編譯…

網絡基礎16--VRRP技術

一、VRRP核心概念定義虛擬路由器冗余協議&#xff08;VRRP&#xff0c;Virtual Router Redundancy Protocol&#xff09;&#xff0c;可以將多個路由器加入到備份組中&#xff0c;形成一臺虛擬路由器&#xff0c;承擔網關功能。RFC 3768標準定義的VRRP是一種容錯協議&#xff0…

最長公共前綴-leetcode

編寫一個函數來查找字符串數組中的最長公共前綴。 如果不存在公共前綴&#xff0c;返回空字符串 “”。 示例 1&#xff1a; 輸入&#xff1a;strs [“flower”,“flow”,“flight”] 輸出&#xff1a;“fl” 示例 2&#xff1a; 輸入&#xff1a;strs [“dog”,“racecar”,…

vs2022:C++安裝opencv

vs2022:C安裝opencv https://opencv.org/releases/ 1.配置包含目錄 2.配置庫目錄 3.配置連接器 4.配置環境變量 5.重新啟動VS2015/VS2017 6.測試 1.配置包含目錄 (頭文件) 2.配置庫目錄&#xff08;dll存放的庫目錄&#xff09; 3.配置連接器(庫) 4.配置環境變量 5.重新啟動VS…

智聯智造:國內新能源汽車品牌AGV小車無線控制系統創新實踐

行業背景&#xff1a;智能制造浪潮下的通信剛需 在全球制造業智能化轉型浪潮中&#xff0c;工業4.0技術已成為提升生產效率與產品質量的核心驅動力。國內某新能源汽車品牌作為智能制造的標桿企業&#xff0c;積極投身自動化設備與智能生產系統的革新。其中&#xff0c;無線控制…

QT6 源,七章對話框與多窗體(8) 消息對話框 QMessageBox :屬性,信號函數,成員函數,以及靜態成員函數,源代碼帶注釋

&#xff08;1&#xff09;消息對話框里&#xff0c;分為通知消息&#xff0c;詢問消息&#xff0c;提醒消息&#xff0c;錯誤消息。可以直接使用本類的靜態函數&#xff0c;簡單。但 QT 的官方說明里&#xff0c;建議使用動態成員函數組件的消息框&#xff0c;而非使用靜態函數…

DAY 7|算法篇——棧與隊列(及重溫數組篇章有感)

今天本來應該寫兩道題把這一章節結束掉&#xff0c;奈何第二題前k個高頻元素需要用的二叉樹相關代碼實在不會寫&#xff08;倒是能看懂&#xff09;等我學完二叉樹再把這道題親自寫一遍吧 今天工作量比較小&#xff0c;準備從第一天的任務開始把題目重新再做一遍 239. 滑動窗…

go語言基礎與進階

&#x1f680; Go語言終極高手之路&#xff1a;從基礎到架構的終極指南 Go語言&#xff0c;以其簡潔的語法、卓越的性能和原生的并發模型&#xff0c;席卷了云原生和后端開發領域。然而&#xff0c;要真正駕馭Go&#xff0c;僅僅停留在會寫if-else和for循環是遠遠不夠的。真正的…

Oracle數據恢復—Oracle數據庫所在分區被刪除后報錯的數據恢復案例

Oracle數據庫數據恢復環境&故障&#xff1a; 一臺服務器上一個分區存放Oracle數據庫數據。由于管理員誤操作不小心刪除了該分區&#xff0c;數據庫報錯&#xff0c;無法使用。 北亞企安數據恢復工程師到達現場后&#xff0c;將故障服務器中所有硬盤以只讀方式進行完整鏡像。…

Prometheus+altermanager搭配釘釘報警

一、Prometheus介紹 Prometheus是一個開源系統監控和警報工具包&#xff0c;最初在 SoundCloud構建。自 2012 年成立以來&#xff0c;許多公司和組織都采用了 Prometheus&#xff0c;該項目擁有非常活躍的開發者和用戶社區。它現在是一個獨立的開源項目&#xff0c;獨立于任何…

【小白量化智能體】應用6:根據通達信指標等生成機器學習Python程序

【小白量化智能體】應用6&#xff1a;根據通達信指標等生成機器學習Python程序 【小白量化智能體】是指能夠自主或半自主地通過與環境的交互來實現目標或任務的計算實體。智能體技術是一個百科全書&#xff0c;又融合了人工智能、計算機科學、心理學和經濟學等多個領域的知識&a…