Spring Boot @Retryable重試策略

在實(shí)際工作中欠拾,重處理是一個(gè)非常常見(jiàn)的場(chǎng)景胰锌,比如:

  • 發(fā)送消息失敗。
  • 調(diào)用遠(yuǎn)程服務(wù)失敗藐窄。
  • 爭(zhēng)搶鎖失敗资昧。

這些錯(cuò)誤可能是因?yàn)榫W(wǎng)絡(luò)波動(dòng)造成的,等待過(guò)后重處理就能成功枷邪。通常來(lái)說(shuō)榛搔,會(huì)用try/catch诺凡,while循環(huán)之類(lèi)的語(yǔ)法來(lái)進(jìn)行重處理东揣,但是這樣的做法缺乏統(tǒng)一性,并且不是很方便腹泌,要多寫(xiě)很多代碼嘶卧。

然而spring-retry卻可以通過(guò)注解,在不入侵原有業(yè)務(wù)邏輯代碼的方式下凉袱,優(yōu)雅的實(shí)現(xiàn)重處理功能芥吟。

一侦铜、@Retryable是什么?

spring系列的spring-retry是另一個(gè)實(shí)用程序模塊钟鸵,可以幫助我們以標(biāo)準(zhǔn)方式處理任何特定操作的重試钉稍。在spring-retry中,所有配置都是基于簡(jiǎn)單注釋的棺耍。

二贡未、使用步驟

1.POM依賴(lài)
<dependency>
  <groupId>org.springframework.retry</groupId>
  <artifactId>spring-retry</artifactId>
 </dependency>
2.啟用@Retryable
@EnableRetry
@SpringBootApplication
public class HelloApplication {

    public static void main(String[] args) {
        SpringApplication.run(HelloApplication.class, args);
    }

}
3.在方法上添加@Retryable
import com.mail.elegant.service.TestRetryService;
import org.springframework.retry.annotation.Backoff;
import org.springframework.retry.annotation.Retryable;
import org.springframework.stereotype.Service;
import java.time.LocalTime;

@Service
public class TestRetryServiceImpl implements TestRetryService {

    @Override
    @Retryable(value = Exception.class,maxAttempts = 3,backoff = @Backoff(delay = 2000,multiplier = 1.5))
    public int test(int code) throws Exception{
        System.out.println("test被調(diào)用,時(shí)間:"+LocalTime.now());
          if (code==0){
              throw new Exception("情況不對(duì)頭!");
          }
        System.out.println("test被調(diào)用,情況對(duì)頭了蒙袍!");

        return 200;
    }
}

來(lái)簡(jiǎn)單解釋一下注解中幾個(gè)參數(shù)的含義:

  • value:拋出指定異常才會(huì)重試
  • include:和value一樣俊卤,默認(rèn)為空,當(dāng)exclude也為空時(shí)害幅,默認(rèn)所有異常
  • exclude:指定不處理的異常
  • maxAttempts:最大重試次數(shù)消恍,默認(rèn)3次
  • backoff:重試等待策略,默認(rèn)使用@Backoff以现,@Backoff的value默認(rèn)為1000L狠怨,我們?cè)O(shè)置為2000L;multiplier(指定延遲倍數(shù))默認(rèn)為0邑遏,表示固定暫停1秒后進(jìn)行重試取董,如果把multiplier設(shè)置為1.5,則第一次重試為2秒无宿,第二次為3秒茵汰,第三次為4.5秒。

當(dāng)重試耗盡時(shí)還是失敗孽鸡,會(huì)出現(xiàn)什么情況呢蹂午?

當(dāng)重試耗盡時(shí),RetryOperations可以將控制傳遞給另一個(gè)回調(diào)彬碱,即RecoveryCallback豆胸。Spring-Retry還提供了@Recover注解,用于@Retryable重試失敗后處理方法巷疼。如果不需要回調(diào)方法晚胡,可以直接不寫(xiě)回調(diào)方法,那么實(shí)現(xiàn)的效果是嚼沿,重試次數(shù)完了后估盘,如果還是沒(méi)成功沒(méi)符合業(yè)務(wù)判斷,就拋出異常骡尽。

4.@Recover
@Recover
public int recover(Exception e, int code){
   System.out.println("回調(diào)方法執(zhí)行G餐住!E氏浮箫踩!");
   //記日志到數(shù)據(jù)庫(kù) 或者調(diào)用其余的方法
    return 400;
}

可以看到傳參里面寫(xiě)的是 Exception e爱态,這個(gè)是作為回調(diào)的接頭暗號(hào)(重試次數(shù)用完了,還是失敗境钟,我們拋出這個(gè)Exception e通知觸發(fā)這個(gè)回調(diào)方法)锦担。

對(duì)于@Recover注解的方法,需要特別注意的是:

  • 方法的返回值必須與@Retryable方法一致
  • 方法的第一個(gè)參數(shù)慨削,必須是Throwable類(lèi)型的吆豹,建議是與@Retryable配置的異常一致,其他的參數(shù)理盆,需要哪個(gè)參數(shù)痘煤,寫(xiě)進(jìn)去就可以了(@Recover方法中有的)
  • 該回調(diào)方法與重試方法寫(xiě)在同一個(gè)實(shí)現(xiàn)類(lèi)里面
5. 注意事項(xiàng)
  • 由于是基于AOP實(shí)現(xiàn),所以不支持類(lèi)里自調(diào)用方法
  • 如果重試失敗需要給@Recover注解的方法做后續(xù)處理猿规,那這個(gè)重試的方法不能有返回值衷快,只能是void
  • 方法內(nèi)不能使用try catch,只能往外拋異常
  • @Recover注解來(lái)開(kāi)啟重試失敗后調(diào)用的方法(注意,需跟重處理方法在同一個(gè)類(lèi)中)姨俩,此注解注釋的方法參數(shù)一定要是@Retryable拋出的異常蘸拔,否則無(wú)法識(shí)別,可以在該方法中進(jìn)行日志處理环葵。
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末调窍,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子张遭,更是在濱河造成了極大的恐慌邓萨,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,000評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件菊卷,死亡現(xiàn)場(chǎng)離奇詭異缔恳,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)洁闰,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,745評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)歉甚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人扑眉,你說(shuō)我怎么就攤上這事纸泄。” “怎么了腰素?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,561評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵聘裁,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我耸弄,道長(zhǎng)咧虎,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,782評(píng)論 1 298
  • 正文 為了忘掉前任计呈,我火速辦了婚禮砰诵,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘捌显。我一直安慰自己茁彭,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,798評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布扶歪。 她就那樣靜靜地躺著理肺,像睡著了一般。 火紅的嫁衣襯著肌膚如雪善镰。 梳的紋絲不亂的頭發(fā)上妹萨,一...
    開(kāi)封第一講書(shū)人閱讀 52,394評(píng)論 1 310
  • 那天,我揣著相機(jī)與錄音炫欺,去河邊找鬼乎完。 笑死,一個(gè)胖子當(dāng)著我的面吹牛品洛,可吹牛的內(nèi)容都是我干的树姨。 我是一名探鬼主播,決...
    沈念sama閱讀 40,952評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼桥状,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼帽揪!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起辅斟,我...
    開(kāi)封第一講書(shū)人閱讀 39,852評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤转晰,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后士飒,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體挽霉,經(jīng)...
    沈念sama閱讀 46,409評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,483評(píng)論 3 341
  • 正文 我和宋清朗相戀三年变汪,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了侠坎。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,615評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡裙盾,死狀恐怖实胸,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情番官,我是刑警寧澤庐完,帶...
    沈念sama閱讀 36,303評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站徘熔,受9級(jí)特大地震影響门躯,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜酷师,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,979評(píng)論 3 334
  • 文/蒙蒙 一讶凉、第九天 我趴在偏房一處隱蔽的房頂上張望染乌。 院中可真熱鬧,春花似錦懂讯、人聲如沸荷憋。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,470評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)勒庄。三九已至,卻和暖如春瘫里,著一層夾襖步出監(jiān)牢的瞬間实蔽,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,571評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工谨读, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留局装,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,041評(píng)論 3 377
  • 正文 我出身青樓漆腌,卻偏偏與公主長(zhǎng)得像贼邓,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子闷尿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,630評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容