Spring Boot 2.x實(shí)戰(zhàn)之StateMachine

本文首發(fā)于個(gè)人網(wǎng)站:Spring Boot 2.x實(shí)戰(zhàn)之StateMachine

Spring StateMachine是一個(gè)狀態(tài)機(jī)框架,在Spring框架項(xiàng)目中,開發(fā)者可以通過簡(jiǎn)單的配置就能獲得一個(gè)業(yè)務(wù)狀態(tài)機(jī),而不需要自己去管理狀態(tài)機(jī)的定義、初始化等過程。今天這篇文章,我們通過一個(gè)案例學(xué)習(xí)下Spring StateMachine框架的用法。

案例介紹

假設(shè)在一個(gè)業(yè)務(wù)系統(tǒng)中,有這樣一個(gè)對(duì)象,它有三個(gè)狀態(tài):草稿、待發(fā)布、發(fā)布完成,針對(duì)這三個(gè)狀態(tài)的業(yè)務(wù)動(dòng)作也比較簡(jiǎn)單,分別是:上線、發(fā)布、回滾。該業(yè)務(wù)狀態(tài)機(jī)如下圖所示。

img

實(shí)戰(zhàn)

接下來(lái),基于上面的業(yè)務(wù)狀態(tài)機(jī)進(jìn)行Spring StateMachine的演示。

  • 創(chuàng)建一個(gè)基礎(chǔ)的Spring Boot工程,在主pom文件中加入Spring StateMachine的依賴:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>
  <parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.2.1.RELEASE</version>
    <relativePath/> <!-- lookup parent from repository -->
  </parent>
  <groupId>online.javaadu</groupId>
  <artifactId>statemachinedemo</artifactId>
  <version>0.0.1-SNAPSHOT</version>
  <name>statemachinedemo</name>
  <description>Demo project for Spring Boot</description>

  <properties>
    <java.version>1.8</java.version>
  </properties>

  <dependencies>
    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter</artifactId>
    </dependency>

    <dependency>
      <groupId>org.projectlombok</groupId>
      <artifactId>lombok</artifactId>
      <optional>true</optional>
    </dependency>
    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-test</artifactId>
      <scope>test</scope>
      <exclusions>
        <exclusion>
          <groupId>org.junit.vintage</groupId>
          <artifactId>junit-vintage-engine</artifactId>
        </exclusion>
      </exclusions>
    </dependency>

    <!--加入spring statemachine的依賴-->
        <dependency>
        <groupId>org.springframework.statemachine</groupId>
        <artifactId>spring-statemachine-core</artifactId>
        <version>2.1.3.RELEASE</version>
      </dependency>
  </dependencies>

  <build>
    <plugins>
      <plugin>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-maven-plugin</artifactId>
      </plugin>
    </plugins>
  </build>

</project>

定義狀態(tài)枚舉和事件枚舉,代碼如下:

/**
* 狀態(tài)枚舉
**/
public enum States {
    DRAFT,
    PUBLISH_TODO,
    PUBLISH_DONE,
}

/**
* 事件枚舉
**/
public enum Events {
    ONLINE,
    PUBLISH,
    ROLLBACK
}
  • 完成狀態(tài)機(jī)的配置,包括:(1)狀態(tài)機(jī)的初始狀態(tài)和所有狀態(tài);(2)狀態(tài)之間的轉(zhuǎn)移規(guī)則
@Configuration
@EnableStateMachine
public class StateMachineConfig extends EnumStateMachineConfigurerAdapter<States, Events> {

    @Override
    public void configure(StateMachineStateConfigurer<States, Events> states) throws Exception {
        states.withStates().initial(States.DRAFT).states(EnumSet.allOf(States.class));
    }

    @Override
    public void configure(StateMachineTransitionConfigurer<States, Events> transitions) throws Exception {
        transitions.withExternal()
            .source(States.DRAFT).target(States.PUBLISH_TODO)
            .event(Events.ONLINE)
            .and()
            .withExternal()
            .source(States.PUBLISH_TODO).target(States.PUBLISH_DONE)
            .event(Events.PUBLISH)
            .and()
            .withExternal()
            .source(States.PUBLISH_DONE).target(States.DRAFT)
            .event(Events.ROLLBACK);
    }
}
  • 定義一個(gè)測(cè)試業(yè)務(wù)對(duì)象,狀態(tài)機(jī)的狀態(tài)轉(zhuǎn)移都會(huì)反映到該業(yè)務(wù)對(duì)象的狀態(tài)變更上
@WithStateMachine
@Data
@Slf4j
public class BizBean {

    /**
     * @see States
     */
    private String status = States.DRAFT.name();

    @OnTransition(target = "PUBLISH_TODO")
    public void online() {
        log.info("操作上線,待發(fā)布. target status:{}", States.PUBLISH_TODO.name());
        setStatus(States.PUBLISH_TODO.name());
    }

    @OnTransition(target = "PUBLISH_DONE")
    public void publish() {
        log.info("操作發(fā)布,發(fā)布完成. target status:{}", States.PUBLISH_DONE.name());
        setStatus(States.PUBLISH_DONE.name());
    }

    @OnTransition(target = "DRAFT")
    public void rollback() {
        log.info("操作回滾,回到草稿狀態(tài). target status:{}", States.DRAFT.name());
        setStatus(States.DRAFT.name());
    }

}
  • 編寫測(cè)試用例,這里我們使用CommandLineRunner接口代替,定義了一個(gè)StartupRunner,在該類的run方法中啟動(dòng)狀態(tài)機(jī)、發(fā)送不同的事件,通過日志驗(yàn)證狀態(tài)機(jī)的流轉(zhuǎn)過程。
public class StartupRunner implements CommandLineRunner {

    @Resource
    StateMachine<States, Events> stateMachine;

    @Override
    public void run(String... args) throws Exception {
        stateMachine.start();
        stateMachine.sendEvent(Events.ONLINE);
        stateMachine.sendEvent(Events.PUBLISH);
        stateMachine.sendEvent(Events.ROLLBACK);
    }
}

在運(yùn)行上述程序后,我們可以在控制臺(tái)中獲得如下輸出,我們執(zhí)行了三個(gè)操作:上線、發(fā)布、回滾,在下圖中也確實(shí)看到了對(duì)應(yīng)的日志。不過我還發(fā)現(xiàn)有一個(gè)意料之外的地方——在啟動(dòng)狀態(tài)機(jī)的時(shí)候,還打印出了一個(gè)日志——“操作回滾,回到草稿狀態(tài). target status:DRAFT”,這里應(yīng)該是狀態(tài)機(jī)設(shè)置初始狀態(tài)的時(shí)候觸發(fā)的。

image-20191110162618938

分析

如上面的實(shí)戰(zhàn)過程所示,使用Spring StateMachine的步驟如下:

  1. 定義狀態(tài)枚舉和事件枚舉
  2. 定義狀態(tài)機(jī)的初始狀態(tài)和所有狀態(tài)
  3. 定義狀態(tài)之間的轉(zhuǎn)移規(guī)則
  4. 在業(yè)務(wù)對(duì)象中使用狀態(tài)機(jī),編寫響應(yīng)狀態(tài)變化的監(jiān)聽器方法

為了將狀態(tài)變更的操作都統(tǒng)一管理起來(lái),我們會(huì)考慮在項(xiàng)目中引入狀態(tài)機(jī),這樣其他的業(yè)務(wù)模塊就和狀態(tài)轉(zhuǎn)移模塊隔離開來(lái)了,其他業(yè)務(wù)模塊也不會(huì)糾結(jié)于當(dāng)前的狀態(tài)是什么,應(yīng)該做什么操作。在應(yīng)用狀態(tài)機(jī)實(shí)現(xiàn)業(yè)務(wù)需求時(shí),關(guān)鍵是業(yè)務(wù)狀態(tài)的分析,只要狀態(tài)機(jī)設(shè)計(jì)得沒問題,具體的實(shí)現(xiàn)可以選擇用Spring StateMachine,也可以自己去實(shí)現(xiàn)一個(gè)狀態(tài)機(jī)。

使用Spring StateMachine的好處在于自己無(wú)需關(guān)心狀態(tài)機(jī)的實(shí)現(xiàn)細(xì)節(jié),只需要關(guān)心業(yè)務(wù)有什么狀態(tài)、它們之間的轉(zhuǎn)移規(guī)則是什么、每個(gè)狀態(tài)轉(zhuǎn)移后真正要進(jìn)行的業(yè)務(wù)操作。

本文完整實(shí)例參見:https://github.com/duqicauc/Spring-Boot-2.x-In-Action/tree/master/statemachinedemo

參考資料

  1. http://blog.didispace.com/spring-statemachine/
  2. https://projects.spring.io/spring-statemachine/#quick-start

本號(hào)專注于后端技術(shù)、JVM問題排查和優(yōu)化、Java面試題、個(gè)人成長(zhǎng)和自我管理等主題,為讀者提供一線開發(fā)者的工作和成長(zhǎng)經(jīng)驗(yàn),期待你能在這里有所收獲。
javaadu
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,461評(píng)論 6 532
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,538評(píng)論 3 417
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,423評(píng)論 0 375
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,991評(píng)論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,761評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,207評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,268評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,419評(píng)論 0 288
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,959評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,782評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,983評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,528評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,222評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,901評(píng)論 1 286
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,678評(píng)論 3 392
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,978評(píng)論 2 374

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