玩轉Spring狀態機

来源:https://www.cnblogs.com/Jcloud/archive/2023/12/20/17915667.html
-Advertisement-
Play Games

本文主要介紹了設計模式中的狀態模式,併在此基礎上介紹了Spring狀態機相關的概念,並根據常見的訂單流轉場景,介紹了Spring狀態機的使用方式。文中如有不當之處,歡迎在評論區批評指正。 ...


說起Spring狀態機,大家很容易聯想到這個狀態機和設計模式中狀態模式的區別是啥呢?沒錯,Spring狀態機就是狀態模式的一種實現,在介紹Spring狀態機之前,讓我們來看看設計模式中的狀態模式。

1. 狀態模式

狀態模式的定義如下:

狀態模式(State Pattern)是一種行為型設計模式,它允許對象在內部狀態發生變化時改變其行為。在狀態模式中,一個對象的行為取決於其當前狀態,而且可以隨時改變這個狀態。狀態模式將對象的狀態封裝在不同的狀態類中,從而使代碼更加清晰和易於維護。當一個對象的狀態改變時,狀態模式會自動更新該對象的行為,而不需要在代碼中手動進行判斷和處理。

通常業務系統中會存在一些擁有狀態的對象,而且這些狀態之間可以進行轉換,並且在不同的狀態下會表現出不同的行為或者不同的功能,比如交通燈控制系統中會存在紅燈、綠燈和黃燈,再比如訂單系統中的訂單會存在已下單、待支付、待發貨、待收貨等狀態,這些狀態會通過不同的行為進行相互轉換,這時候在系統設計時就可以使用狀態模式。

下麵是狀態模式類圖:

可以看到狀態模式主要包含三種類型的角色:

1、上下文(**Context**)角色:封裝了狀態的實例,負責維護狀態實例,並將請求委托給當前的狀態對象。

2、抽象狀態(**State**)角色:定義了表示不同狀態的介面,並封裝了該狀態下的行為。所有具體狀態都實現這個介面。

3、具體狀態(**Concrete State**)角色:具體實現了抽象狀態角色的介面,並封裝了該狀態下的行為。

下麵是使用狀態模式實現紅綠燈狀態變更的一個簡單案例:

抽象狀態類:

/**
 * @description: 抽象狀態類
 */
public abstract class MyState {
    abstract void handler();
}

具體狀態類A

/**
 * @description: 具體狀態A
 */
public class RedLightState extends MyState{

    @Override
    void handler() {
        System.out.println("紅燈停");
    }
}



具體狀態類B

/**
 * @description: 具體狀態B
 */
public class GreenLightState extends MyState{

    @Override
    void handler() {
        System.out.println("綠燈行");
    }
}



環境類:維護當前狀態對象,並提供了切換狀態的方法。

/**
 * @description: 環境類
 */
public class MyContext {

    private MyState state;

    public void setState(MyState state) {
        this.state = state;
    }

    public void handler() {
        state.handler();
    }
}

測試類

/**
 * @description: 測試狀態模式
 */
public class TestStateModel {
    public static void main(String[] args) {
        MyContext myContext = new MyContext();

        RedLightState redLightState = new RedLightState();
        GreenLightState greenLightState = new GreenLightState();

        myContext.setState(redLightState);
        myContext.handler(); //紅燈停

        myContext.setState(greenLightState);
        myContext.handler(); //綠燈行
    }
}

下麵是對應的執行結果

可以發現,使用狀態模式中的狀態類在一定程度上也消除了if-else邏輯校驗,看到這裡, 有些人可能會有疑問:狀態模式和策略模式的區別是什麼呢?

狀態模式更關註對象在不同狀態的行為和狀態之間的流轉,而策略模式更關註對象不同策略的選擇。

上面我們介紹了設計模式中的狀態模式,接下來我們來看看Spring狀態機。

2. Spring狀態機

狀態機,也就是 State Machine ,不是指一臺實際機器,而是指一個數學模型 。說白了,就是指一張狀態轉換圖。 狀態機是狀態模式的一種應用,相當於上下文角色的一個升級版。在工作流或游戲等各種系統中有大量使用,如各種工作流引擎,它幾乎是狀態機的子集和實現,封裝狀態的變化規則。Spring也提供了一個很好的解決方案。Spring中的組件名稱就叫作狀態機(StateMachine)。狀態機幫助開發者簡化狀態控制的開發過程,讓狀態機結構更加層次化。

通過定義,我們很容易分析得到狀態機應當具備一下幾個要素:

  1. 當前狀態:也就是狀態流轉的起始狀態。

  2. 觸發事件:引起狀態之間流轉的一些列動作。

  3. 響應函數:觸發事件到下一個狀態之間的規則。

  4. 目標狀態:狀態流轉的目標狀態。

對於組件化的狀態機,當前使用較多的主要是兩種:一種是Spring 狀態機,一種是COLA狀態機,這兩種狀態機的對比如下表所示:

Spring 狀態機 COLA 狀態機
API 調用 使用 Reactive 的 Mono、Flux 方式進行 API 調用 同步的 API 調用,如果有需要也可以將方法通過 消息隊列、定時任務、多線程等方式進行非同步調用
代碼量 core 包 284 個介面和類 36 個介面和類
生態 非常豐富 較為貧瘠
定製化難度 困難 簡單

可以看到,Spring狀態機鎖提供的內容較為豐富,當然對於自定義的支持就不如COLA狀態機好,如果對自定義的需求比較高,那建議使用COLA狀態機。

本文以Spring狀態機為例,展示如何在業務系統中使用狀態機。

為了便於大家瞭解Spring狀態機的實現原理和使用方式以及其提供的功能,下麵列出了官方文檔和源碼,感興趣的同學可以閱讀閱讀。

官方文檔: https://docs.spring.io/spring-statemachine/docs/4.0.0/reference/index.html#statemachine-config-states

源代碼: https://github.com/spring-projects/spring-statemachine

3. Spring狀態機實現訂單狀態流轉

對於狀態模式,Spring封裝好了一個組件,就叫狀態機(StateMachine)。Spring狀態機可以幫助我們開發者簡化狀態控制的開發過程,讓狀態機結構更加層次化。下麵用Spring狀態機模擬一個訂單狀態流轉的過程。

3.1 環境準備

首先,如果要使用spring狀態機,需要引入對應的jar包,這裡我的springboot版本是:2.2.1.RELEASE

<dependency>
    <groupId>org.springframework.statemachine</groupId>
    <artifactId>spring-statemachine-core</artifactId>
    <version>${springboot.version}</version>
</dependency>

下麵是簡化的訂單的定義,以及訂單狀態和訂單轉換行為的枚舉

/**
 * @description: 模擬訂單類
 */
@Data
public class Order {
    private Long orderId;
    private OrderStatusEnum orderStatus;
}

/**
 * @description: 訂單狀態
 */
public enum OrderStatusEnum {
    // 待支付
    WAIT_PAYMENT,
    // 待發貨
    WAIT_DELIVER,
    // 待收貨
    WAIT_RECEIVE,
    // 完成
    FINISH;
}

/**
 * @description:訂單狀態轉換行為
 */
public enum OrderStatusChangeEventEnum {
    //支付
    PAYED,
    //發貨
    DELIVERY,
    //收貨
    RECEIVED;
}

3.2 構造訂單狀態機

在引入jar包之後,需要構建一個針對訂單狀態流轉的狀態機

訂單狀態機配置類如下:

/**
 * @description: 訂單狀態機
 */
@Configuration
@EnableStateMachine
public class OrderStatusMachineConfig extends StateMachineConfigurerAdapter<OrderStatusEnum, OrderStatusChangeEventEnum> {

    /**
     * 配置狀態
     */
    @Override
    public void configure(StateMachineStateConfigurer<OrderStatusEnum, OrderStatusChangeEventEnum> states) throws Exception {
        states.withStates()
                .initial(OrderStatusEnum.WAIT_PAYMENT)
                .end(OrderStatusEnum.FINISH)
                .states(EnumSet.allOf(OrderStatusEnum.class));
    }

    /**
     * 配置狀態轉換事件關係
     */
    @Override
    public void configure(StateMachineTransitionConfigurer<OrderStatusEnum, OrderStatusChangeEventEnum> transitions) throws Exception {
        transitions.withExternal().source(OrderStatusEnum.WAIT_PAYMENT).target(OrderStatusEnum.WAIT_DELIVER)
                .event(OrderStatusChangeEventEnum.PAYED)
                .and()
                .withExternal().source(OrderStatusEnum.WAIT_DELIVER).target(OrderStatusEnum.WAIT_RECEIVE)
                .event(OrderStatusChangeEventEnum.DELIVERY)
                .and()
                .withExternal().source(OrderStatusEnum.WAIT_RECEIVE).target(OrderStatusEnum.FINISH)
                .event(OrderStatusChangeEventEnum.RECEIVED);
    }
}

3.3 編寫狀態機監聽器

監聽狀態變更事件,完成狀態轉換。

/**
 * @description: 狀態監聽
 */
@Component
@WithStateMachine
@Transactional
public class OrderStatusListener {
    @OnTransition(source = "WAIT_PAYMENT", target = "WAIT_DELIVER")
    public boolean payTransition(Message message) {
        Order order = (Order) message.getHeaders().get("order");
        order.setOrderStatus(OrderStatusEnum.WAIT_DELIVER);
        System.out.println("支付,狀態機反饋信息:" + message.getHeaders().toString());
        return true;
    }

    @OnTransition(source = "WAIT_DELIVER", target = "WAIT_RECEIVE")
    public boolean deliverTransition(Message message) {
        Order order = (Order) message.getHeaders().get("order");
        order.setOrderStatus(OrderStatusEnum.WAIT_RECEIVE);
        System.out.println("發貨,狀態機反饋信息:" + message.getHeaders().toString());
        return true;
    }

    @OnTransition(source = "WAIT_RECEIVE", target = "FINISH")
    public boolean receiveTransition(Message message) {
        Order order = (Order) message.getHeaders().get("order");
        order.setOrderStatus(OrderStatusEnum.FINISH);
        System.out.println("收貨,狀態機反饋信息:" + message.getHeaders().toString());
        return true;
    }

}

3.4 編寫訂單服務類

模擬對訂單的一些業務操作

/**
 * @description: 訂單服務
 */
@Service
public class OrderServiceImpl implements OrderService {

    @Resource
    private StateMachine<OrderStatusEnum, OrderStatusChangeEventEnum> orderStateMachine;

    private long id = 1L;

    private Map<Long, Order> orders = Maps.newConcurrentMap();

    @Override
    public Order create() {
        Order order = new Order();
        order.setOrderStatus(OrderStatusEnum.WAIT_PAYMENT);
        order.setOrderId(id++);
        orders.put(order.getOrderId(), order);
        System.out.println("訂單創建成功:" + order.toString());
        return order;
    }

    @Override
    public Order pay(long id) {
        Order order = orders.get(id);
        System.out.println("嘗試支付,訂單號:" + id);
        Message message = MessageBuilder.withPayload(OrderStatusChangeEventEnum.PAYED).
                setHeader("order", order).build();
        if (!sendEvent(message)) {
            System.out.println(" 支付失敗, 狀態異常,訂單號:" + id);
        }
        return orders.get(id);
    }

    @Override
    public Order deliver(long id) {
        Order order = orders.get(id);
        System.out.println(" 嘗試發貨,訂單號:" + id);
        if (!sendEvent(MessageBuilder.withPayload(OrderStatusChangeEventEnum.DELIVERY)
                .setHeader("order", order).build())) {
            System.out.println(" 發貨失敗,狀態異常,訂單號:" + id);
        }
        return orders.get(id);
    }

    @Override
    public Order receive(long id) {
        Order order = orders.get(id);
        System.out.println(" 嘗試收貨,訂單號:" + id);
        if (!sendEvent(MessageBuilder.withPayload(OrderStatusChangeEventEnum.RECEIVED)
                .setHeader("order", order).build())) {
            System.out.println(" 收貨失敗,狀態異常,訂單號:" + id);
        }
        return orders.get(id);
    }


    @Override
    public Map<Long, Order> getOrders() {
        return orders;
    }

    /**
     * 發送狀態轉換事件
     * @param message
     * @return
     */
    private synchronized boolean sendEvent(Message<OrderStatusChangeEventEnum> message) {
        boolean result = false;
        try {
            orderStateMachine.start();
            result = orderStateMachine.sendEvent(message);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (Objects.nonNull(message)) {
                Order order = (Order) message.getHeaders().get("order");
                if (Objects.nonNull(order) && Objects.equals(order.getOrderStatus(), OrderStatusEnum.FINISH)) {
                    orderStateMachine.stop();
                }
            }
        }
        return result;
    }
}

3.5 測試入口

這裡編寫一個controller模擬c端用戶請求,為了便於展示,這裡使用一個測試方法完成所有的操作

@RestController
public class OrderController {

    @Resource
    private OrderService orderService;

    @RequestMapping("/testOrderStatusChange")
    public String testOrderStatusChange(){
        orderService.create();
        orderService.create();
        orderService.pay(1L);
        orderService.deliver(1L);
        orderService.receive(1L);
        orderService.pay(2L);
        orderService.deliver(2L);
        orderService.receive(2L);
        System.out.println("全部訂單狀態:" + orderService.getOrders());
        return "success";
    }

}

下麵是對應的執行結果

可以看到spring狀態機很好的控制了訂單在各個狀態之間的流轉。

4. 思考與總結

思考:針對狀態機的特點,還有其他思路實現一個狀態機嗎?下麵是一些常規思路,如果還有其他方法歡迎在評論區留言。

1. 消息隊列方式

訂單狀態的流轉可以通過MQ發佈一個事件,消費者根據業務條件把訂單狀態進行流轉,可以根據不同的事件發送到不同的Topic。

2. 定時任務驅動

每隔一段時間啟動一下job,根據特定的狀態從資料庫中拿對應的訂單記錄,然後判斷訂單是否有條件到達下一個狀態。

3. 規則引擎方式

業務團隊可以在規則引擎里編寫一系列的狀態及其對應的轉換規則,由規則引擎根據已經載入的規則對輸入數據進行解析,根據解析的結果執行相應的動作,完成狀態流轉。

總結:

本文主要介紹了設計模式中的狀態模式,併在此基礎上介紹了Spring狀態機相關的概念,並根據常見的訂單流轉場景,介紹了Spring狀態機的使用方式。文中如有不當之處,歡迎在評論區批評指正。

5. 參考內容

https://docs.spring.io/spring-statemachine/docs/4.0.0/reference/index.html#statemachine-config-states

https://cloud.tencent.com/developer/article/2198477?areaId=106001

https://cloud.tencent.com/developer/article/2360708?areaId=106001

https://juejin.cn/post/7087064901553750030

https://my.oschina.net/u/4090830/blog/10092135

https://juejin.cn/post/7267506576448929811

作者:京東科技 孫揚威

來源:京東雲開發者社區 轉載請註明來源


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 隨著人工智慧技術的不斷發展,阿裡體育等IT大廠,推出的“樂動力”、“天天跳繩”AI運動APP,讓雲上運動會、線上運動會、健身打卡、AI體育指導等概念空前火熱。那麼,能否將這些在APP成功應用的場景搬上小程式,分享這些概念的紅利呢?本系列文章就帶您一步一步從零開始開發一個AI運動小程式,本系列文章將使 ...
  • 這章內容詳細地介紹了文件上傳和下載的實現過程。文件上傳涉及前端頁面、Controller 方法和配置修改,其中前端頁面通過表單的提交方式和enctype屬性設置來實現文件上傳,而後端的 Controller 方法則通過接收 MultipartFile 類型的參數來處理上傳的文件,並將文件保存到伺服器... ...
  • C 語言中的運算符 運算符用於對變數和值進行操作。 在下麵的示例中,我們使用 + 運算符將兩個值相加: int myNum = 100 + 50; 雖然 + 運算符通常用於將兩個值相加,就像上面的示例一樣,它還可以用於將變數和值相加,或者將變數和另一個變數相加: int sum1 = 100 + 5 ...
  • Redis 全文搜索是依賴於 Redis 官方提供的 RediSearch 來實現的。RediSearch 提供了一種簡單快速的方法對 hash 或者 json 類型數據的任何欄位建立二級索引,然後就可以對被索引的 hash 或者 json 類型數據欄位進行搜索和聚合操作。 這裡我們把被索引的 ha ...
  • 目錄前言QT小記1. 菜單欄、工具欄、狀態欄2. 自定義的對話框3. 任務管理器4. 鏈接資料庫mysql,sqlite5. Widgets Gallery Example 代碼學習:999.ControlsQT-For-Python1. DemoQT-Quick1. HelloWorld2. 簡單 ...
  • AVL樹 AVL樹是一種自平衡二叉搜索樹。在這種樹中,任何節點的兩個子樹的高度差被嚴格控制在1以內。這確保了樹的平衡,從而保證了搜索、插入和刪除操作的高效性。AVL樹是由Georgy Adelson-Velsky和Evgenii Landis在1962年發明的,因此得名(Adelson-Velsky ...
  • Qt 是一個跨平臺C++圖形界面開發庫,利用Qt可以快速開發跨平臺窗體應用程式,在Qt中我們可以通過拖拽的方式將不同組件放到指定的位置,實現圖形化開發極大的方便了開發效率,本章將重點介紹`TabWidget`標簽組件的常用方法及靈活運用。`QTabWidget` 是Qt中用於實現標簽頁(tabbed... ...
  • JavaSE學習思維導圖 目錄1 Java語言概述1.1 Java 概述1.2 Java 語言簡史1.3 Java 之父1.4 Java 技術體系平臺2 Java 開發環境搭建2.1 JDK JRE2.2 JDK版本的選擇2.3 JDK 的下載2.4 JDK 的安裝2.5 配置path環境變數2.5 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...