Spring Boot 揭秘与实战(七) 实用技术篇 - StateMachine 状态机机制
文章目录
Spring StateMachine 让状态机结构更加层次化,可以帮助开发者简化状态机的开发过程。
之前,我们使用二维数组实现状态机机制,现在,我们来用 Spring StateMachine 进行改造。
环境依赖
修改 POM 文件,添加 spring-statemachine-core 依赖。
- <dependency>
- <groupId>org.springframework.statemachine</groupId>
- <artifactId>spring-statemachine-core</artifactId>
- <version>1.2.0.RELEASE</version>
- </dependency>
状态和事件
现在,我以用户注册为案例,来讲解状态和事件之间的状态机机制。
状态枚举
注册有哪些状态呢,我们来想想,应该有4个状态:未连接、已连接、注册中、已注册。
- public enum RegStatusEnum {
- // 未连接
- UNCONNECTED,
- // 已连接
- CONNECTED,
- // 注册中
- REGISTERING,
- // 已注册
- REGISTERED;
- }
事件枚举
相对应的,存在几个核心事件:连接、注册、注册成功、注册失败、注销。
- public enum RegEventEnum {
- // 连接
- CONNECT,
- // 注册
- REGISTER,
- // 注册成功
- REGISTER_SUCCESS,
- // 注册失败
- REGISTER_FAILED,
- // 注销
- UN_REGISTER;
- }
状态机配置
- @Configuration
- @EnableStateMachine
- public class StateMachineConfig extends EnumStateMachineConfigurerAdapter<RegStatusEnum, RegEventEnum> {
- }
@EnableStateMachine注解,标识启用 Spring StateMachine 状态机功能。
初始化状态机状态
我们需要初始化状态机的状态。
- @Override
- public void configure(StateMachineStateConfigurer<RegStatusEnum, RegEventEnum> states) throws Exception {
- states.withStates()
- // 定义初始状态
- .initial(RegStatusEnum.UNCONNECTED)
- // 定义状态机状态
- .states(EnumSet.allOf(RegStatusEnum.class));
- }
其中,initial(RegStatusEnum.UNCONNECTED) 定义了初始状态是未连接状态。states(EnumSet.allOf(RegStatusEnum.class)) 定义了定义状态机中存在的所有状态。
初始化状态迁移事件
我们需要初始化当前状态机有哪些状态事件。
- @Override
- public void configure(StateMachineTransitionConfigurer<RegStatusEnum, RegEventEnum> transitions)
- throws Exception {
- transitions
- // 1.连接事件
- // 未连接 -> 已连接
- .withExternal()
- .source(RegStatusEnum.UNCONNECTED)
- .target(RegStatusEnum.CONNECTED)
- .event(RegEventEnum.CONNECT)
- .and()
- // 2.注册事件
- // 已连接 -> 注册中
- .withExternal()
- .source(RegStatusEnum.CONNECTED)
- .target(RegStatusEnum.REGISTERING)
- .event(RegEventEnum.REGISTER)
- .and()
- // 3.注册成功事件
- // 注册中 -> 已注册
- .withExternal()
- .source(RegStatusEnum.REGISTERING)
- .target(RegStatusEnum.REGISTERED)
- .event(RegEventEnum.REGISTER_SUCCESS)
- .and()
- // 5.注销事件
- // 已连接 -> 未连接
- .withExternal()
- .source(RegStatusEnum.CONNECTED)
- .target(RegStatusEnum.UNCONNECTED)
- .event(RegEventEnum.UN_REGISTER)
- .and()
- // 注册中 -> 未连接
- .withExternal()
- .source(RegStatusEnum.REGISTERING)
- .target(RegStatusEnum.UNCONNECTED)
- .event(RegEventEnum.UN_REGISTER)
- .and()
- // 已注册 -> 未连接
- .withExternal()
- .source(RegStatusEnum.REGISTERED)
- .target(RegStatusEnum.UNCONNECTED)
- .event(RegEventEnum.UN_REGISTER)
- ;
- }
这里,我以连接事件为案例,其中 source 指定原始状态,target 指定目标状态,event 指定触发事件。
因此,下面的状态就很好理解了,即当发生连接事件时,从未连接状态变更为已连接状态。
- // 未连接 -> 已连接
- .withExternal()
- .source(RegStatusEnum.UNCONNECTED)
- .target(RegStatusEnum.CONNECTED)
- .event(RegEventEnum.CONNECT)
状态监听器
Spring StateMachine 提供了注解配置实现方式,所有 StateMachineListener 接口中定义的事件都能通过注解的方式来进行配置实现。
- @WithStateMachine
- public class StateMachineEventConfig {
- @OnTransition(source = "UNCONNECTED", target = "CONNECTED")
- public void connect() {
- System.out.println("///////////////////");
- System.out.println("连接事件, 未连接 -> 已连接");
- System.out.println("///////////////////");
- }
- @OnTransition(source = "CONNECTED", target = "REGISTERING")
- public void register() {
- System.out.println("///////////////////");
- System.out.println("注册事件, 已连接 -> 注册中");
- System.out.println("///////////////////");
- }
- @OnTransition(source = "REGISTERING", target = "REGISTERED")
- public void registerSuccess() {
- System.out.println("///////////////////");
- System.out.println("注册成功事件, 注册中 -> 已注册");
- System.out.println("///////////////////");
- }
- @OnTransition(source = "REGISTERED", target = "UNCONNECTED")
- public void unRegister() {
- System.out.println("///////////////////");
- System.out.println("注销事件, 已注册 -> 未连接");
- System.out.println("///////////////////");
- }
- }
这里,我仍然以连接事件为案例,@OnTransition 中 source 指定原始状态,target 指定目标状态,当事件触发时将会被监听到从而调用 connect() 方法。
总结
Spring StateMachine 让状态机结构更加层次化,可以帮助开发者简化状态机的开发过程。
我们来回顾下几个核心步骤
- 定义状态枚举。
- 定义事件枚举。
- 定义状态机配置,设置初始状态,以及状态与事件之间的关系。
- 定义状态监听器,当状态变更时,触发方法。
源代码
相关示例完整代码: springboot-action
(完)
如果觉得我的文章对你有帮助,请随意打赏。

- 版权声明:本文由 梁桂钊 发表于 梁桂钊的博客
- 转载声明:自由转载-非商用-非衍生-保持署名(创意共享3.0许可证),非商业转载请注明作者及出处,商业转载请联系作者本人。
- 文章标题:Spring Boot 揭秘与实战(七) 实用技术篇 - StateMachine 状态机机制
- 文章链接:http://blog.720ui.com/2017/springboot_07_othercore_statemachine/
Spring Boot 揭秘与实战(七) 实用技术篇 - StateMachine 状态机机制的更多相关文章
- Spring Boot 揭秘与实战(七) 实用技术篇 - 异步任务
文章目录 1. Spring Boot 集成异步任务 2. 单发服务模式 3. 请求应答模式 4. 源代码 Spring 对异步任务具有很好的支持.这篇文章,我们透过 Spring Boot 来讲解下 ...
- Spring Boot 揭秘与实战(七) 实用技术篇 - Java Mail 发送邮件
文章目录 1. Spring Boot 集成 Java Mail 2. 单元测试 3. 源代码 Spring 对 Java Mail 有很好的支持.因此,Spring Boot 也提供了自动配置的支持 ...
- Spring Boot 揭秘与实战(七) 实用技术篇 - FreeMarker 模板引擎
文章目录 1. FreeMaker 代替 JSP 作为页面渲染 2. 生成静态文件 3. 扩展阅读 4. 源代码 Spring Boot 提供了很多模板引擎的支持,例如 FreeMarker.Thym ...
- Spring Boot 揭秘与实战 附录 - Spring Boot 公共配置
Spring Boot 公共配置,配置 application.properties/application.yml 文件中. 摘自:http://docs.spring.io/spring-boot ...
- Spring Boot 揭秘与实战 自己实现一个简单的自动配置模块
文章目录 1. 实战的开端 – Maven搭建 2. 参数的配置 - 属性参数类 3. 真的很简单 - 简单的服务类 4. 自动配置的核心 - 自动配置类 5. spring.factories 不要 ...
- Spring Boot 揭秘与实战 源码分析 - 工作原理剖析
文章目录 1. EnableAutoConfiguration 帮助我们做了什么 2. 配置参数类 – FreeMarkerProperties 3. 自动配置类 – FreeMarkerAutoCo ...
- Spring Boot 揭秘与实战 源码分析 - 开箱即用,内藏玄机
文章目录 1. 开箱即用,内藏玄机 2. 总结 3. 源代码 Spring Boot提供了很多”开箱即用“的依赖模块,那么,Spring Boot 如何巧妙的做到开箱即用,自动配置的呢? 开箱即用,内 ...
- Spring Boot 揭秘与实战(九) 应用监控篇 - 自定义监控端点
文章目录 1. 继承 AbstractEndpoint 抽象类 2. 创建端点配置类 3. 运行 4. 源代码 Spring Boot 提供的端点不能满足我们的业务需求时,我们可以自定义一个端点. 本 ...
- Spring Boot 揭秘与实战(九) 应用监控篇 - HTTP 健康监控
文章目录 1. 内置 HealthIndicator 监控检测 2. 自定义 HealthIndicator 监控检测 3. 源代码 Health 信息是从 ApplicationContext 中所 ...
随机推荐
- 牛客练习赛32-D-MST+tarjin割边
链接:https://ac.nowcoder.com/acm/contest/272/D来源:牛客网 题目描述 小p和他的朋友约定好去游乐场游玩,但是他们到了游乐场后却互相找不到对方了. 游乐场可以看 ...
- GitHub学习三-远程版本库更新与提交
1.远程版本库更新 一般来说,将本地与远程相关联之后,首先将数据从远程更新下来再上传比较好. 输入 git pull origin master 如果新建版本库的话勾选了初始化包含readme.md, ...
- SQL 查询语句
4.2 单表查询 4.2.1 列名(表名)的别名(as 可以不加) 给列名取别名既可以加 as 也可以不加. (2008 - Sage.lower(Sdept)等可计算但无列名,需要指定列名) 原列名 ...
- 【LeetCode】数组移除元素
链表等复杂数据结构用多了,简单的数组操作也不能遗忘! 1. 给定一个有序数组,移除所有重复元素并返回新的数组长度,不能分配额外数组的内存空间. e.g. 给定输入的数组 = [1,1,2],函数应当返 ...
- [luogu P3384] [模板]树链剖分
[luogu P3384] [模板]树链剖分 题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点 ...
- 安天透过北美DDoS事件解读IoT设备安全——Mirai的主要感染对象是linux物联网设备,包括:路由器、网络摄像头、DVR设备,入侵主要通过telnet端口进行流行密码档暴力破解,或默认密码登陆,下载DDoS功能的bot,运行控制物联网设备
安天透过北美DDoS事件解读IoT设备安全 安天安全研究与应急处理中心(安天CERT)在北京时间10月22日下午启动高等级分析流程,针对美国东海岸DNS服务商Dyn遭遇DDoS攻击事件进行了跟进分析. ...
- Greys Java在线问题诊断工具
摘要: 线上系统为何经常出错?数据库为何屡遭黑手?业务调用为何频频失败?连环异常堆栈案,究竟是那次调用所为? 数百台服务器意外雪崩背后又隐藏着什么?是软件的扭曲还是硬件的沦丧? 走进科学带你了解Gre ...
- 在springboot中验证表单信息(六)
构建工程 创建一个springboot工程,由于用到了 web .thymeleaf.validator.el,引入相应的起步依赖和依赖,代码清单如下: 1 2 3 4 5 6 7 8 9 10 11 ...
- SpringMVC中JSP页面显示为源码
@RequestMapping(value = "login") public ModelAndView login(ModelAndView mav) throws Except ...
- API设计和命名
1.前言 对于前端开发而言,肯定会和API打交道,大家也都会想过怎么设计自己的API.优秀的 API 之于代码,就如良好内涵对于每个人.好的 API 不但利于使用者理解,开发时也会事半功倍,后期维护更 ...