一個註解@Recover搞定醜陋的迴圈重試代碼

来源:https://www.cnblogs.com/mmzs/archive/2022/03/30/16078880.html
-Advertisement-
Play Games

在實際項目中其中一部分邏輯可能會因為調用了外部服務或者等待鎖等情況下出現不可預料的異常,在這個時候我們可能需要對調用這部分邏輯進行重試,代碼裡面主要就是使用for迴圈寫一大坨重試的邏輯,各種硬編碼,各種辣眼睛的補丁。 ...


使用背景

在實際項目中其中一部分邏輯可能會因為調用了外部服務或者等待鎖等情況下出現不可預料的異常,在這個時候我們可能需要對調用這部分邏輯進行重試,代碼裡面主要就是使用for迴圈寫一大坨重試的邏輯,各種硬編碼,各種辣眼睛的補丁。

特別是針對重試的邏輯,到處都有。所以我決定用一個重試組件spring-retry優化一波。它的出現,解決掉這部分醜陋的代碼!

這個組件的源碼地址如下:https://github.com/spring-projects/spring-retry

廢話不多說,直接上代碼吧!

開始上代碼

首先引入依賴:

<dependency>
    <groupId>org.springframework.retry</groupId>
    <artifactId>spring-retry</artifactId>
    <version>1.3.2</version>
</dependency>

由於該組件是依賴於 AOP 給你的,所以還需要引入這個依賴(如果你其他 jar 包中引用過了,當然也就不需要再次引用了):

<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 {

    /**
     * 測試service
     *
     * @return
     */
    List<Integer> testService();
}

Service層具體實現

@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("錯誤的返回");
        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);
    }
}

@Retryable註解重要屬性解析

  • recover: 此類中用於恢復的方法的名稱。方法必須用 {@link Recover} 註釋標記。
  • value: 可重試的異常類型。包括()的同義詞。預設為空(如果 excludes 也為空,則重試所有異常)。
  • exclude: 不可重試的異常類型。預設為空(如果包含也為空,則重試所有異常)。如果 include 為空但 excludes 不是,則重試所有未排除的異常
  • maxAttempts: 方法重試調用次數,預設3次
  • backoff: 指定用於重試此操作的其他屬性

@backoff註解

  • value:重試之間間隔時間
  • delay:重試之間的等待時間(以毫秒為單位)
  • maxDelay:重試之間的最大等待時間(以毫秒為單位)
  • multiplier:指定延遲的倍數
  • delayExpression:重試之間的等待時間表達式
  • maxDelayExpression:重試之間的最大等待時間表達式
  • multiplierExpression:指定延遲的倍數表達式
  • random:隨機指定延遲時間

@Recover註解

主要作用是標記方法為一個重試方法的補償方法!!!

註意事項

  • 方法重試依賴於 spring 註入,所以調用的方法的類必須是被spring管理的,然後通過 @Autowired 或 @Resource 引入使用,不然不會生效
  • 方法重試的前提是方法拋出了異常,在方法執行出現了異常且沒有被捕獲的情況下重試
  • 方法重試需要在方法上面加上註解 @Retryable
  • 方法重試的補償方法上面必須攜帶@Recover註解
  • @Recover方法需要和@Retryable方法在同一個類中才能生效@Recover方法(@Recover方法在父類中也可以生效)
  • 使用@Retryable註解,如果類中沒有被@Recover標示的方法,無論是否使用 recover 屬性都拋出原有異常
  • 使用@Retryable註解同時 recover 屬性不是空,如果類中有@Recover標示的方法,但是標示的方法不是 recover 指定的方法,拋出ExhaustedRetryException異常
  • 使用@Retryable註解同時 recover 屬性不是空,同時方法有註解@Recover,但是補償方法的參數不是當前異常或者異常的父類,拋出ExhaustedRetryException 異常
  • 使用@Retryable註解不使用 recover 屬性,如果類中被@Recover標示的方法有和原方法返回值一樣的,使用當前被@Recover標示的方法(此時方法參數可隨意,但是不能是除開當前異常的類及父類的異常類)
如果,您希望更容易地發現我的博客,不妨收藏關註一下
原創公眾號:Java學習之道
個人博客 : www.mmzsblog.cn

喜歡就推薦一下,因為你的參與是我在寫作道路上的最強動力。
本文版權歸作者淼淼之森和博客園共有,歡迎轉載。
但須在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 框架看起來就像是宗教(或者說是政治):每一個框架都假裝自己為開發者提供瞭解決方案,但每一個又都不一樣。它們每一個都聲稱可以為應用程式提供最好的前景,但關於哪一個真正名副其實的爭論又不絕於耳。每一個框架都要求你遵循特定的規則,它們之間可能有相似之處,但要從一個框架轉換到另一個框架總是很難。所以沒有什麼... ...
  • 屬性的簡潔表示法 ES6 允許在大括弧裡面直接寫入變數和函數,作為對象的屬性和方法。這樣的書寫更加簡潔。 const foo = 'bar'; const baz = { foo }; console.log(baz); // { foo: 'bar' } function f(name, age) ...
  • 一、冒泡排序 原理:相鄰兩元素之間兩兩比較,比較出大值進行賦值互換,再依次與相鄰的元素比較,層層遞進。#互換元素位置,相互賦值。 時間複雜度:最好O(n),最差O(n^2) 1、比較相鄰的兩個元素,如果前一個比後一個大,則交換位置。2、比較完第一輪的時候,最後一個元素是最大的元素。3、這時候最後一個 ...
  • 魔術方法(特定時機自動觸發) __init__ 構造方法 觸發時機:實例化對象,初始化的時候觸發 功能:為對象添加成員 參數:參數不固定,至少一個self參數 返回值:無 # (1) 基本語法 class MyClass(): def __init__(self): print("構造方法被觸發 . ...
  • 代理模式是什麼 代理模式是一種結構型設計模式, 讓你能提供真實服務對象的替代品給客戶端使用。 代理接收客戶端的請求併進行一些處理 (訪問控制和緩存等), 然後再將請求傳遞給服務對象。 為什麼用代理模式 在某些情況下客戶類不想或者不能訪問目標對象,這時候就可以使用代理類訪問。 代理模式怎麼實現 pac ...
  • import 導入模塊或包 文件就是一個模塊,文件夾就是一個包文件夾裡面可以有很多文件,就相當於包中有好多的模塊. import 模塊或者包(包是文件夾,模塊是文件)模塊不會被重覆導入,引入一次終生受益 '''調用的時候: 模塊.變數 模塊.函數 模塊.類''' (1) 模塊.變數 print(my ...
  • 1.標題 //標題一共六個級別 # 一級標題 ## 二級標題 ### 三級標題 #### 四級標題 ##### 五級標題 ###### 六級標題 #一級標題 ##二級標題 ###三級標題 ####四級標題 #####五級標題 六級標題 2.字體 **粗體** *斜體* ***粗體加斜體*** ~~刪 ...
  • 內容概要 序列化器介紹 Serializer的使用 基本使用(序列化) 欄位類型 欄位參數 序列化 定製序列化的欄位 反序列化 反序列化之新增 反序列化之修改 反序列化之局部和全局鉤子 ModelSerializer 模型類序列化器 ModelSerializer 額外添加參數 內容詳細 序列化器介 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...