通过实例学习SpringStateMachine之Washer

2023-10-10 07:20

本文主要是介绍通过实例学习SpringStateMachine之Washer,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景介绍

本系列通过学习SpringStateMachine中附带的10余个Sample来学习SpringStateMachine中的各个概念和用法。项目是使用的分支为2.2.0.RELEASE。项目参考文档也是2.2.0.RELEASE。

Washer简介

Washer是对洗衣机的描述。通过建立洗衣机的的状态机来演示如何使用history来实现状态的恢复。 例如甩干的过程中突然按下了暂停按钮。如果继续点击恢复,那么会恢复到原来的状态。

下图是washer状态机的描述。这里利用到了HISTORY这个伪状态来实现历史状态的恢复。当启动机器后,RUNNING下有3个子状态。如果当前在子状态DRYING下,且产生了CUTPOWER事件产,那么RUNNING会转换到POWEROFF。如果在次产生RESTOREPOWER事件,那么会恢复到RUNNING状态下的子状态。从而实现了状态恢复的演示。
在这里插入图片描述

Washer 依赖

项目在实现上述功能时,需要依赖springshell,官方给出的demo[1]使用了spring-shell1.2,本文将其改为spring-shell 2.0.0.RELEASE。

<dependency><groupId>org.springframework.statemachine</groupId><artifactId>spring-statemachine-core</artifactId><version>2.2.0.RELEASE</version></dependency><dependency><groupId>org.springframework.shell</groupId><artifactId>spring-shell-starter</artifactId><version>2.0.0.RELEASE</version></dependency>

Washer 实现

Washer依然是一个分层状态机。整个状态机的状态如下:

public enum States {RUNNING, HISTORY, END,WASHING, RINSING, DRYING,POWEROFF
}

RUNNING状态下包含一个子层,内有3个状态。而状态恢复就是当状态转换到某个子层后,能再恢复到变换前的子状态。

状态机的事件如下:

public enum Events {RINSE, DRY, STOP,RESTOREPOWER, CUTPOWER
}

首先我们需要配置状态。为了配置HISTORY这个伪状态,我们用到了history方法,同时指定了状态类型为History.SHALLOW。WASHING、RINSING、RINSING、HISTORY均是RUNNING的子状态。

@Override
public void configure(StateMachineStateConfigurer<States, Events> states)throws Exception {states.withStates().initial(States.RUNNING).state(States.POWEROFF).end(States.END).and().withStates().parent(States.RUNNING).initial(States.WASHING).state(States.RINSING).state(States.DRYING).history(States.HISTORY, History.SHALLOW);
}

接着我们添加转换。核心点就是子状态机无论内部子状态是什么,在CUTPOWER事件产生后都能够转换到POWEROFF状态。

其次是POWEROFF到HISTORY状态转换,由于HISTORY是一个RUNNING子状态机中的伪状态。转换到HISOTRY后,RUNNING内部子状态会恢复到从RUNNING转换到POWEROFF前的状态。

@Override
public void configure(StateMachineTransitionConfigurer<States, Events> transitions)throws Exception {transitions.withExternal().source(States.WASHING).target(States.RINSING).event(Events.RINSE).and().withExternal().source(States.RINSING).target(States.DRYING).event(Events.DRY).and().withExternal().source(States.RUNNING).target(States.POWEROFF).event(Events.CUTPOWER).and().withExternal().source(States.POWEROFF).target(States.HISTORY).event(Events.RESTOREPOWER).and().withExternal().source(States.RUNNING).target(States.END).event(Events.STOP);
}

为了观察效果,我们增加一下Action来打印输出,进行观察。完整的配置如下:

import org.springframework.context.annotation.Configuration;
import org.springframework.statemachine.StateContext;
import org.springframework.statemachine.action.Action;
import org.springframework.statemachine.config.EnableStateMachine;
import org.springframework.statemachine.config.EnumStateMachineConfigurerAdapter;
import org.springframework.statemachine.config.builders.StateMachineStateConfigurer;
import org.springframework.statemachine.config.builders.StateMachineTransitionConfigurer;
import org.springframework.statemachine.config.configurers.StateConfigurer;@Configuration
@EnableStateMachine
public class StateMachineConfigextends EnumStateMachineConfigurerAdapter<States, Events> {private Action<States, Events> runningAction() {return new Action<States, Events>() {@Overridepublic void execute(StateContext<States, Events> context) {System.out.println("runningAction");}};}private Action<States, Events> powerOffAction() {return new Action<States, Events>() {@Overridepublic void execute(StateContext<States, Events> context) {System.out.println("powerOffAction");}};}private Action<States, Events> rinsingAction() {return new Action<States, Events>() {@Overridepublic void execute(StateContext<States, Events> context) {System.out.println("rinsingAction");}};}private Action<States, Events> washingAction() {return new Action<States, Events>() {@Overridepublic void execute(StateContext<States, Events> context) {System.out.println("washingAction");}};}private Action<States, Events> dryingAction() {return new Action<States, Events>() {@Overridepublic void execute(StateContext<States, Events> context) {System.out.println("drying");}};}@Overridepublic void configure(StateMachineStateConfigurer<States, Events> states)throws Exception {states.withStates().initial(States.RUNNING,runningAction()).state(States.POWEROFF,powerOffAction()).end(States.END).and().withStates().parent(States.RUNNING).initial(States.WASHING,washingAction()).state(States.RINSING,rinsingAction()).state(States.DRYING,dryingAction()).history(States.HISTORY, StateConfigurer.History.DEEP);}@Overridepublic void configure(StateMachineTransitionConfigurer<States, Events> transitions)throws Exception {transitions.withExternal().source(States.WASHING).target(States.RINSING).event(Events.RINSE).and().withExternal().source(States.RINSING).target(States.DRYING).event(Events.DRY).and().withExternal().source(States.RUNNING).target(States.POWEROFF).event(Events.CUTPOWER).and().withExternal().source(States.POWEROFF).target(States.HISTORY).event(Events.RESTOREPOWER).and().withExternal().source(States.RUNNING).target(States.END).event(Events.STOP);}
}

最后我们通过命令行发送命令即可观察RUNNING内部子状态的恢复。


Entry state RUNNING
Entry state WASHING
State machine startedsm>sm event RINSE
Exit state WASHING
Entry state RINSING
Event RINSE sendsm>sm event DRY
Exit state RINSING
Entry state DRYING
Event DRY sendsm>sm event CUTPOWER
Exit state DRYING
Exit state RUNNING
Entry state POWEROFF
Event CUTPOWER sendsm>sm event RESTOREPOWER
Exit state POWEROFF
Entry state RUNNING
Entry state WASHING
Entry state DRYING
Event RESTOREPOWER send

总结

在分层状态机中,一个子状态机发生状态变化后,如果状态又转移到子状态机上,为了恢复子状态机中的内部状态,可以利用伪状态HISTORY来实现状态恢复。

参考

[1]washer doc, https://docs.spring.io/spring-statemachine/docs/2.2.0.RELEASE/reference/#statemachine-examples-washer
[2]washer source code,https://github.com/spring-projects/spring-statemachine/tree/2.2.0.RELEASE/spring-statemachine-samples/washer/src/main/java/demo/washer

这篇关于通过实例学习SpringStateMachine之Washer的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/178904

相关文章

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析

《Spring组件实例化扩展点之InstantiationAwareBeanPostProcessor使用场景解析》InstantiationAwareBeanPostProcessor是Spring... 目录一、什么是InstantiationAwareBeanPostProcessor?二、核心方法解

java String.join()方法实例详解

《javaString.join()方法实例详解》String.join()是Java提供的一个实用方法,用于将多个字符串按照指定的分隔符连接成一个字符串,这一方法是Java8中引入的,极大地简化了... 目录bVARxMJava String.join() 方法详解1. 方法定义2. 基本用法2.1 拼接

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio

Linux lvm实例之如何创建一个专用于MySQL数据存储的LVM卷组

《Linuxlvm实例之如何创建一个专用于MySQL数据存储的LVM卷组》:本文主要介绍使用Linux创建一个专用于MySQL数据存储的LVM卷组的实例,具有很好的参考价值,希望对大家有所帮助,... 目录在Centos 7上创建卷China编程组并配置mysql数据目录1. 检查现有磁盘2. 创建物理卷3. 创

Java List排序实例代码详解

《JavaList排序实例代码详解》:本文主要介绍JavaList排序的相关资料,Java排序方法包括自然排序、自定义排序、Lambda简化及多条件排序,实现灵活且代码简洁,文中通过代码介绍的... 目录一、自然排序二、自定义排序规则三、使用 Lambda 表达式简化 Comparator四、多条件排序五、

Java实例化对象的​7种方式详解

《Java实例化对象的​7种方式详解》在Java中,实例化对象的方式有多种,具体取决于场景需求和设计模式,本文整理了7种常用的方法,文中的示例代码讲解详细,有需要的可以了解下... 目录1. ​new 关键字(直接构造)​2. ​反射(Reflection)​​3. ​克隆(Clone)​​4. ​反序列化

重新对Java的类加载器的学习方式

《重新对Java的类加载器的学习方式》:本文主要介绍重新对Java的类加载器的学习方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍1.1、简介1.2、符号引用和直接引用1、符号引用2、直接引用3、符号转直接的过程2、加载流程3、类加载的分类3.1、显示

Python解决雅努斯问题实例方案详解

《Python解决雅努斯问题实例方案详解》:本文主要介绍Python解决雅努斯问题实例方案,雅努斯问题是指AI生成的3D对象在不同视角下出现不一致性的问题,即从不同角度看物体时,物体的形状会出现不... 目录一、雅努斯简介二、雅努斯问题三、示例代码四、解决方案五、完整解决方案一、雅努斯简介雅努斯(Janu