使用注解@Recover優(yōu)化丑陋的循環(huán)詳解
1使用背景
在實(shí)際項(xiàng)目中其中一部分邏輯可能會(huì)因?yàn)檎{(diào)用了外部服務(wù)或者等待鎖等情況下出現(xiàn)不可預(yù)料的異常,在這個(gè)時(shí)候我們可能需要對(duì)調(diào)用這部分邏輯進(jìn)行重試,代碼里面主要就是使用for循環(huán)寫一大坨重試的邏輯,各種硬編碼,各種辣眼睛的補(bǔ)丁。
特別是針對(duì)重試的邏輯,到處都有。所以我決定用一個(gè)重試組件spring-retry優(yōu)化一波。它的出現(xiàn),解決掉這部分丑陋的代碼!
這個(gè)組件的源碼地址如下:https://github.com/spring-projects/spring-retry

廢話不多說(shuō),直接上代碼吧!
2開始上代碼
首先引入依賴:
<dependency>
<groupId>org.springframework.retry</groupId>
<artifactId>spring-retry</artifactId>
<version>1.3.2</version>
</dependency>
由于該組件是依賴于 AOP 給你的,所以還需要引入這個(gè)依賴(如果你其他 jar 包中引用過了,當(dāng)然也就不需要再次引用了):
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
<version>2.6.1</version>
</dependency>
開啟重試:
@SpringBootApplication
@EnableRetry
public class ApplicationStarter {
public static void main(String[] args) {
SpringApplication.run(ApplicationStarter.class);
}
}
Controller層
@RestController
public class TestController {
@Autowired
private IRecursiveCallService recursiveCallService;
@GetMapping("test2")
public Object test2() {
return recursiveCallService.testService();
}
}
Service層
public interface IRecursiveCallService {
/**
* 測(cè)試service
*
* @return
*/
List<Integer> testService();
}
Service層具體實(shí)現(xiàn)
@Service
public class RecursiveCallServiceImpl implements IRecursiveCallService {
@Override
@Retryable(recover = "testService3")
public List<Integer> testService() {
System.out.println("到此一游!");
System.out.println(1 / 0);
return null;
}
@Recover
public List<String> testService1() {
System.out.println("錯(cuò)誤的返回");
return Collections.singletonList("S");
}
@Recover
public List<Integer> testService2(String i) {
System.out.println("正確的返回");
return Collections.singletonList(1);
}
@Recover
public List<Integer> testService3() {
System.out.println("正確的返回2");
return Collections.singletonList(2);
}
}
3@Retryable注解重要屬性解析
- recover: 此類中用于恢復(fù)的方法的名稱。方法必須用 {@link Recover} 注釋標(biāo)記。
- value: 可重試的異常類型。包括()的同義詞。默認(rèn)為空(如果 excludes 也為空,則重試所有異常)。
- exclude: 不可重試的異常類型。默認(rèn)為空(如果包含也為空,則重試所有異常)。如果 include 為空但 excludes 不是,則重試所有未排除的異常
- maxAttempts: 方法重試調(diào)用次數(shù),默認(rèn)3次
- backoff: 指定用于重試此操作的其他屬性
4@backoff注解
- value:重試之間間隔時(shí)間
- delay:重試之間的等待時(shí)間(以毫秒為單位)
- maxDelay:重試之間的最大等待時(shí)間(以毫秒為單位)
- multiplier:指定延遲的倍數(shù)
- delayExpression:重試之間的等待時(shí)間表達(dá)式
- maxDelayExpression:重試之間的最大等待時(shí)間表達(dá)式
- multiplierExpression:指定延遲的倍數(shù)表達(dá)式
- random:隨機(jī)指定延遲時(shí)間
5@Recover注解
主要作用是標(biāo)記方法為一個(gè)重試方法的補(bǔ)償方法?。?!
6注意事項(xiàng)
方法重試依賴于 spring 注入,所以調(diào)用的方法的類必須是被spring管理的,然后通過 @Autowired 或 @Resource 引入使用,不然不會(huì)生效
方法重試的前提是方法拋出了異常,在方法執(zhí)行出現(xiàn)了異常且沒有被捕獲的情況下重試
方法重試需要在方法上面加上注解 @Retryable
方法重試的補(bǔ)償方法上面必須攜帶@Recover注解
@Recover方法需要和@Retryable方法在同一個(gè)類中才能生效@Recover方法(@Recover方法在父類中也可以生效)
使用@Retryable注解,如果類中沒有被@Recover標(biāo)示的方法,無(wú)論是否使用 recover 屬性都拋出原有異常
使用@Retryable注解同時(shí) recover 屬性不是空,如果類中有@Recover標(biāo)示的方法,但是標(biāo)示的方法不是 recover 指定的方法,拋出ExhaustedRetryException異常
使用@Retryable注解同時(shí) recover 屬性不是空,同時(shí)方法有注解@Recover,但是補(bǔ)償方法的參數(shù)不是當(dāng)前異?;蛘弋惓5母割悾瑨伋鯡xhaustedRetryException 異常
使用@Retryable注解不使用 recover 屬性,如果類中被@Recover標(biāo)示的方法有和原方法返回值一樣的,使用當(dāng)前被@Recover標(biāo)示的方法(此時(shí)方法參數(shù)可隨意,但是不能是除開當(dāng)前異常的類及父類的異常類)
總結(jié)
到此這篇關(guān)于使用注解@Recover優(yōu)化丑陋的循環(huán)的文章就介紹到這了,更多相關(guān)注解@Recover優(yōu)化循環(huán)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決IDEA修改 .vmoptions 文件后導(dǎo)致無(wú)法啟動(dòng)的問題
這篇文章主要介紹了解決IDEA修改 .vmoptions 文件后導(dǎo)致無(wú)法啟動(dòng)的問題,需要的朋友可以參考下2020-12-12
java編程實(shí)現(xiàn)國(guó)際象棋棋盤
這篇文章主要為大家詳細(xì)介紹了java編程實(shí)現(xiàn)國(guó)際象棋棋盤,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-05-05
SpringBoot四大神器之Auto onfiguration的使用
本文主要介紹了SpringBoot四大神器之Auto Configuration,springboot auto configuration的本質(zhì)就是自動(dòng)配置spring的各種bean。感興趣的可以了解一下2021-10-10
Java利用OSHI實(shí)現(xiàn)獲取機(jī)器的硬件信息
OSHI(Operating System and Hardware Information)是一個(gè)開源的Java庫(kù),用于獲取操作系統(tǒng)和硬件的詳細(xì)信息,下面我們就來(lái)看看他的具體使用吧2024-11-11
Java聊天室之實(shí)現(xiàn)一個(gè)服務(wù)器與多個(gè)客戶端通信
這篇文章主要為大家詳細(xì)介紹了Java簡(jiǎn)易聊天室之實(shí)現(xiàn)一個(gè)服務(wù)器與多個(gè)客戶端通信,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,需要的可以了解一下2022-10-10
SpringBoot3集成iText實(shí)現(xiàn)PDF導(dǎo)出功能
不知道小伙伴們?cè)陧?xiàng)目中有沒有遇到過導(dǎo)出 PDF 的需求,小編在之前的 tienchin 項(xiàng)目中有一個(gè)合同導(dǎo)出的功能,需要將文檔導(dǎo)出為PDF,將文檔導(dǎo)出為 PDF 有很多方案,不同方案的優(yōu)缺點(diǎn)也各不相同,今天小編就和大家演示一個(gè),感興趣的小伙伴跟著小編一起來(lái)看看吧2024-10-10

