![731ecf3c-df3e-11ed-bfe3-dac502259ad0.jpg](https://file1.elecfans.com//web2/M00/9A/B0/wKgaomTnkvuAQavzAABWtQ6XghE345.jpg)
前言
在實際工作中,重處理是一個非常常見的場景,比如:
- 發送消息失敗。
- 調用遠程服務失敗。
- 爭搶鎖失敗。
這些錯誤可能是因為網絡波動造成的,等待過后重處理就能成功。通常來說,會用try/catch
,while
循環之類的語法來進行重處理,但是這樣的做法缺乏統一性,并且不是很方便,要多寫很多代碼。然而spring-retry
卻可以通過注解,在不入侵原有業務邏輯代碼的方式下,優雅的實現重處理功能。
基于 Spring Boot + MyBatis Plus + Vue & Element 實現的后臺管理系統 + 用戶小程序,支持 RBAC 動態權限、多租戶、數據權限、工作流、三方登錄、支付、短信、商城等功能
- 項目地址:https://github.com/YunaiV/ruoyi-vue-pro
- 視頻教程:https://doc.iocoder.cn/video/
@Retryable是什么?
spring系列的spring-retry
是另一個實用程序模塊,可以幫助我們以標準方式處理任何特定操作的重試。在spring-retry
中,所有配置都是基于簡單注釋的。
基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 實現的后臺管理系統 + 用戶小程序,支持 RBAC 動態權限、多租戶、數據權限、工作流、三方登錄、支付、短信、商城等功能
使用步驟
POM依賴
<dependency>
<groupId>org.springframework.retrygroupId>
<artifactId>spring-retryartifactId>
dependency>
啟用@Retryable
@EnableRetry
@SpringBootApplication
publicclassHelloApplication{
publicstaticvoidmain(String[]args){
SpringApplication.run(HelloApplication.class,args);
}
}
在方法上添加@Retryable
importcom.mail.elegant.service.TestRetryService;
importorg.springframework.retry.annotation.Backoff;
importorg.springframework.retry.annotation.Retryable;
importorg.springframework.stereotype.Service;
importjava.time.LocalTime;
@Service
publicclassTestRetryServiceImplimplementsTestRetryService{
@Override
@Retryable(value=Exception.class,maxAttempts=3,backoff=@Backoff(delay=2000,multiplier=1.5))
publicinttest(intcode)throwsException{
System.out.println("test被調用,時間:"+LocalTime.now());
if(code==0){
thrownewException("情況不對頭!");
}
System.out.println("test被調用,情況對頭了!");
return200;
}
}
來簡單解釋一下注解中幾個參數的含義:
-
value
:拋出指定異常才會重試 -
include
:和value一樣,默認為空,當exclude也為空時,默認所有異常 -
exclude
:指定不處理的異常 -
maxAttempts
:最大重試次數,默認3次 -
backoff
:重試等待策略,默認使用@Backoff
,@Backoff
的value默認為1000L,我們設置為2000L;multiplier
(指定延遲倍數)默認為0,表示固定暫停1秒后進行重試,如果把multiplier
設置為1.5,則第一次重試為2秒,第二次為3秒,第三次為4.5秒。
當重試耗盡時還是失敗,會出現什么情況呢?
當重試耗盡時,RetryOperations
可以將控制傳遞給另一個回調,即RecoveryCallback
。Spring-Retry
還提供了@Recover
注解,用于@Retryable重試失敗后處理方法。如果不需要回調方法,可以直接不寫回調方法,那么實現的效果是,重試次數完了后,如果還是沒成功沒符合業務判斷,就拋出異常。
@Recover
@Recover
publicintrecover(Exceptione,intcode){
System.out.println("回調方法執行!!!!");
//記日志到數據庫或者調用其余的方法
return400;
}
可以看到傳參里面寫的是 Exception e
,這個是作為回調的接頭暗號(重試次數用完了,還是失敗,我們拋出這個Exception e
通知觸發這個回調方法)。對于@Recover
注解的方法,需要特別注意的是:
-
方法的返回值必須與
@Retryable
方法一致 -
方法的第一個參數,必須是Throwable類型的,建議是與
@Retryable
配置的異常一致,其他的參數,需要哪個參數,寫進去就可以了(@Recover
方法中有的) - 該回調方法與重試方法寫在同一個實現類里面
注意事項
- 由于是基于AOP實現,所以不支持類里自調用方法
-
如果重試失敗需要給
@Recover
注解的方法做后續處理,那這個重試的方法不能有返回值,只能是void -
方法內不能使用
try catch
,只能往外拋異常 -
@Recover
注解來開啟重試失敗后調用的方法(注意,需跟重處理方法在同一個類中),此注解注釋的方法參數一定要是@Retryable
拋出的異常,否則無法識別,可以在該方法中進行日志處理。
總結
本篇主要簡單介紹了Springboot中的Retryable
的使用,主要的適用場景和注意事項,當需要重試的時候還是很有用的。
審核編輯 :李倩
-
參數
+關注
關注
11文章
1860瀏覽量
32431 -
spring
+關注
關注
0文章
340瀏覽量
14391
原文標題:求求你們了,別再重復造輪子了,一個 Spring 注解輕松搞定循環重試功能!
文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
Spring Boot如何實現異步任務
使用Spring自定義注解的實現
教你如何用一個注解搞定Spring Boot接口防刷
淺談Spring事務底層原理
從源碼層面深度剖析Spring循環依賴
Spring Web MVC注解
![<b class='flag-5'>Spring</b> Web MVC<b class='flag-5'>注解</b>](https://file1.elecfans.com/web2/M00/81/FF/wKgaomQvjouAEB3mAAEJjN1OpNQ018.jpg)
Spring Bean相關的4個注解及使用方法
![<b class='flag-5'>Spring</b> Bean相關的4<b class='flag-5'>個</b><b class='flag-5'>注解</b>及使用方法](https://file1.elecfans.com/web2/M00/81/FF/wKgaomQvjviAWyDeAADCHy-wd84212.jpg)
Spring Dependency Inject與Bean Scops注解
![<b class='flag-5'>Spring</b> Dependency Inject與Bean Scops<b class='flag-5'>注解</b>](https://file1.elecfans.com/web2/M00/81/FF/wKgaomQvjzuANjfUAAFTbw7zzes878.jpg)
Spring中@Component注解是怎么實現的
![<b class='flag-5'>Spring</b>中@Component<b class='flag-5'>注解</b>是怎么實現的](https://file.elecfans.com/web2/M00/9D/F3/pYYBAGQvxC2ADEcoAAAxhbkime4164.png)
評論