Spring-Retry(重試機制)解讀
Spring-Retry(重試機制)
在實際工作中,重處理是一個非常常見的場景,比如:
發(fā)送消息失敗。調用遠程服務失敗。爭搶鎖失敗。
這些錯誤可能是因為網(wǎng)絡波動造成的,等待過后重處理就能成功。
通常來說,會用 try/catch
, while
循環(huán)之類的語法來進行重處理,但是這樣的做法缺乏統(tǒng)一性,并且不是很方便,要多寫很多代碼。
然而 spring-retry
卻可以通過注解,在不入侵原有業(yè)務邏輯代碼的方式下,優(yōu)雅的實現(xiàn)重處理功能。
一、@Retryable是什么?
spring系列的 spring-retry
是另一個實用程序模塊,可以幫助我們以標準方式處理任何特定操作的重試。
在 spring-retry
中,所有配置都是基于簡單注釋的。
二、使用步驟
1.POM依賴
基于AOP實現(xiàn),因此還需引入aop相關的依賴
<dependency> <groupId>org.springframework.retry</groupId> <artifactId>spring-retry</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency>
2.啟用 @Retryable
@EnableRetry @SpringBootApplication public class HelloApplication { public static void main(String[] args) { SpringApplication.run(HelloApplication.class, args); } }
3.在方法上添加 @Retryable
1.創(chuàng)建測試controller
@RestController public class TestRetryController { @Resource private TestRetryService testRetryService; @RequestMapping(value = "/test") public int test(int code) throws Exception { return testRetryService.testRetry(code); } }
2.創(chuàng)建service接口
public interface TestRetryService { int testRetry(int code) throws Exception; }
3.創(chuàng)建service實現(xiàn)類并添加@Retryable
@Service public class TestRetryServiceImpl implements TestRetryService { /** * value:拋出指定異常才會重試 * include:和value一樣,默認為空,當exclude也為空時,默認所有異常 * exclude:指定不處理的異常 * maxAttempts:最大重試次數(shù),默認3次 * backoff:重試等待策略, * 默認使用@Backoff,@Backoff的value默認為1000L,我們設置為2000; 以毫秒為單位的延遲(默認 1000) * multiplier(指定延遲倍數(shù))默認為0,表示固定暫停1秒后進行重試,如果把multiplier設置為1.5,則第一次重試為2秒,第二次為3秒,第三次為4.5秒。 * @param code * @return * @throws Exception */ @Override @Retryable(value = Exception.class,maxAttempts = 3,backoff = @Backoff(delay = 2000,multiplier = 1.5)) public int testRetry(int code) throws Exception{ System.out.println("test被調用,時間:"+ LocalTime.now()); if (code==0){ throw new Exception("情況不對頭!"); } System.out.println("test被調用,情況對頭了!"); return 200; } /** * Spring-Retry還提供了@Recover注解,用于@Retryable重試失敗后處理方法。 * 如果不需要回調方法,可以直接不寫回調方法,那么實現(xiàn)的效果是,重試次數(shù)完了后,如果還是沒成功沒符合業(yè)務判斷,就拋出異常。 * 可以看到傳參里面寫的是 Exception e,這個是作為回調的接頭暗號(重試次數(shù)用完了,還是失敗,我們拋出這個Exception e通知觸發(fā)這個回調方法)。 * 注意事項: * 方法的返回值必須與@Retryable方法一致 * 方法的第一個參數(shù),必須是Throwable類型的,建議是與@Retryable配置的異常一致,其他的參數(shù),需要哪個參數(shù),寫進去就可以了(@Recover方法中有的) * 該回調方法與重試方法寫在同一個實現(xiàn)類里面 * * 由于是基于AOP實現(xiàn),所以不支持類里自調用方法 * 如果重試失敗需要給@Recover注解的方法做后續(xù)處理,那這個重試的方法不能有返回值,只能是void * 方法內不能使用try catch,只能往外拋異常 * @Recover注解來開啟重試失敗后調用的方法(注意,需跟重處理方法在同一個類中),此注解注釋的方法參數(shù)一定要是@Retryable拋出的異常,否則無法識別,可以在該方法中進行日志處理。 * @param e * @param code * @return */ @Recover public int recover(Exception e, int code){ System.out.println("回調方法執(zhí)行!?。?!"); //記日志到數(shù)據(jù)庫 或者調用其余的方法 System.out.println("異常信息:"+e.getMessage()); return 400; } }
來簡單解釋一下注解中幾個參數(shù)的含義:
value
:拋出指定異常才會重試include
:和value一樣,默認為空,當exclude也為空時,默認所有異常exclude
:指定不處理的異常maxAttempts
:最大重試次數(shù),默認3次backoff
:重試等待策略,默認使用@Backoff
,@Backoff
的value默認為1000(單位毫秒),我們設置為2000;multiplier
(指定延遲倍數(shù))默認為0,表示固定暫停1秒后進行重試,如果把multiplier
設置為1.5,則第一次重試為2秒,第二次為3秒,第三次為4.5秒。
當重試耗盡時還是失敗,會出現(xiàn)什么情況呢?
當重試耗盡時, RetryOperations
可以將控制傳遞給另一個回調,即 RecoveryCallback
。
Spring-Retry
還提供了 @Recover
注解,用于@Retryable重試失敗后處理方法。如果不需要回調方法,可以直接不寫回調方法,那么實現(xiàn)的效果是,重試次數(shù)完了后,如果還是沒成功沒符合業(yè)務判斷,就拋出異常。
@Recover public int recover(Exception e, int code){ System.out.println("回調方法執(zhí)行!?。?!"); //記日志到數(shù)據(jù)庫 或者調用其余的方法 System.out.println("異常信息:"+e.getMessage()); return 400; }
4.測試效果
這里我們傳入的code=0因此走異常的分支,由于我們添加@Retryable注解并設置了相關的參數(shù),程序會按照配置進行重試,重試都失敗之后會調用@Recover修飾的方法,最終返回400給瀏覽器
控制臺輸出結果
總結
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
mybatis連接PGSQL中對于json和jsonb的處理方法
在使用PostgreSQL數(shù)據(jù)庫時,將表字段設置為jsonb格式可以存儲JSON數(shù)據(jù),本文給大家介紹mybatis連接PGSQL中對于json和jsonb的處理方法,感興趣的朋友一起看看吧2024-11-11IDEA如何切換數(shù)據(jù)庫版本mysql5或mysql8
本文介紹了如何將IntelliJ IDEA從MySQL5切換到MySQL8的詳細步驟,包括下載MySQL8、安裝、配置、停止舊服務、啟動新服務以及更改密碼等2025-01-01如何使用MyBatis框架實現(xiàn)增刪改查(CRUD)操作
本文主要介紹了如何使用MyBatis框架實現(xiàn)增刪改查(CRUD)操作。首先介紹了MyBatis框架的基本概念和使用方法,然后分別介紹了如何使用MyBatis實現(xiàn)增刪改查操作。最后,通過一個簡單的示例演示了如何使用MyBatis框架實現(xiàn)CRUD操作。2023-05-05Java使用application.property讀取文件里面的值
本文通過實例代碼給大家介紹了Java使用application.property讀取文件里面的值,需要的朋友可以參考下2018-10-10spring 整合 mybatis 中數(shù)據(jù)源的幾種配置方式(總結篇)
因為spring 整合mybatis的過程中, 有好幾種整合方式,尤其是數(shù)據(jù)源那塊,經(jīng)??吹讲灰粯拥呐渲梅绞?,總感覺有點亂,所以今天有空總結下,感興趣的朋友跟隨腳本之家小編一起學習吧2018-05-05spring boot使用logback實現(xiàn)多環(huán)境日志配置詳解
這篇文章主要介紹了spring boot使用logback實現(xiàn)多環(huán)境日志配置詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-08-08jsp頁面中獲取servlet請求中的參數(shù)的辦法詳解
在JAVA WEB應用中,如何獲取servlet請求中的參數(shù),本文講解了jsp頁面中獲取servlet請求中的參數(shù)的辦法2018-03-03