设计模式之状态机模式

一、状态机模式介绍

状态机模式(State Machine Pattern)是一种用于描述对象行为的软件设计模式,属于行为型设计模式。在状态机模式中,对象的行为取决于其内部状态,并且在不同的状态下,对象可能会有不同的行为。这种模式通常涉及定义一组状态以及状态之间的转换规则,从而实现对对象行为的精确控制。

1、状态机模式的基本概念

状态机类图:

状态机模式主要包含以下几个要素:

  1. 状态(State):表示对象所处的特定状态。每个状态都定义了对象在该状态下的行为。
  2. 上下文(Context):上下文是包含状态机的对象。它维护了当前状态,并在状态之间的转换发生时更新状态。
  3. 转换(Transition):描述了对象从一个状态转移到另一个状态的过程。它通常受到一些条件或触发事件的影响。
  4. 动作(Action):动作是状态转换期间可能执行的操作或行为。这些动作可以是更新状态、执行计算、发送消息等。

2、状态机模式的特点

  1. 清晰的状态管理:通过明确定义系统的所有可能状态以及在这些状态之间的转换,帮助开发者清晰地管理和跟踪系统的状态。
  2. 简化复杂逻辑:将复杂的条件分支逻辑转换为状态图,使得逻辑更加直观易懂。
  3. 易于维护和扩展:状态机的结构使得对系统的修改和扩展变得更加容易。通过添加新的状态和转换规则,可以轻松适应需求的变化。
  4. 提高可测试性:由于状态机的行为是预定义的,因此可以更系统地进行测试,有助于确保系统的正确性和可靠性。

3、状态机模式的应用场景

状态机模式在多种应用场景中都有广泛的应用,包括但不限于以下几个方面:

  1. 游戏开发:用于游戏中的角色状态管理,如角色的移动、攻击、防御等状态。
  2. 嵌入式系统:用于描述设备的状态和状态转移,如自动售货机、电梯控制等。
  3. 网络通信:用于管理网络连接的状态,如连接建立、数据传输、连接关闭等。
  4. UI设计:用于管理UI元素的交互逻辑,如按钮的点击状态、表单的输入验证状态等。

二、状态机模式的实现例子

下面是一个简单的Java状态机模式例子。在这个例子中,我们将模拟一个简单的交通信号灯系统,它有三种状态:红灯(Red)、绿灯(Green)、黄灯(Yellow)。每个状态都有一个特定的行为,即打印出当前灯的颜色,并且每个状态都可以转换到下一个状态。

首先,我们定义一个TrafficLightState接口,它表示交通信号灯的所有可能状态:

public interface TrafficLightState {  void change(TrafficLightContext context);  
}

然后,我们为每种状态实现这个接口:

public class RedLightState implements TrafficLightState {  @Override  public void change(TrafficLightContext context) {  System.out.println("红灯亮");  context.setState(new GreenLightState());  }  
}  public class GreenLightState implements TrafficLightState {  @Override  public void change(TrafficLightContext context) {  System.out.println("绿灯亮");  context.setState(new YellowLightState());  }  
}  public class YellowLightState implements TrafficLightState {  @Override  public void change(TrafficLightContext context) {  System.out.println("黄灯亮");  context.setState(new RedLightState());  }  
}

接下来,我们定义TrafficLightContext类,它包含了当前的状态,并且提供了一个方法来改变状态:

public class TrafficLightContext {  private TrafficLightState state;  public TrafficLightContext() {  this.state = new RedLightState();  }  public void setState(TrafficLightState state) {  this.state = state;  }  public void change() {  state.change(this);  }  
}

最后,我们创建一个客户端来模拟交通信号灯的状态变化:

public class Client {  public static void main(String[] args) {  TrafficLightContext context = new TrafficLightContext();  // 模拟状态变化  context.change();  context.change();  context.change();  context.change();  context.change();  }  
}

在这个例子中,TrafficLightContext类维护了当前的交通信号灯状态,并且提供了一个change方法来改变状态。每个状态类都实现了TrafficLightState接口,并且在其change方法中定义了下一个状态。客户端通过调用TrafficLightContextchange方法来模拟交通信号灯的状态变化。 

现实中的状态转变,比这个复杂多了,一个状态节点可能有多个action操作,类似订单状态的流转,无法直接套用状态机模式,应该有记录状态变迁的全景图(核心要素:当前状态,当前状态可选动作,动作执行后状态迁移)。

以下是一个简单的例子,记录当前状态,可选动作及执行动作后状态的一个例子,用于约束状态的变迁,。

package demo;import java.util.HashMap;
import java.util.Map;public class StateMachine {/*** key:当前状态 ,value: Map<String, String>,可选的动作及后续状态,key:xxxAction,value:nextState*/private Map<String, Map<String, String>> stateMap;public StateMachine() {stateMap = new HashMap<String, Map<String, String>>();// 初始状态可以执行的Action及执行后的动作Map<String, String> initActionMap = new HashMap<String, String>();initActionMap.put("runAction", "run");stateMap.put("init", initActionMap);// run状态下可以执行的动作Map<String, String> runActionMap = new HashMap<String, String>();runActionMap.put("stopAction", "stop");runActionMap.put("sleepAction", "sleep");runActionMap.put("endAction", "end");stateMap.put("run", runActionMap);// sleep状态下可以执行的动作Map<String, String> sleepActionMap = new HashMap<String, String>();sleepActionMap.put("runAction", "run");stateMap.put("sleep", sleepActionMap);// stop状态下可以执行的动作Map<String, String> stopActionMap = new HashMap<String, String>();stopActionMap.put("runAction", "run");stateMap.put("stop", stopActionMap);}public String getNextState(String curState, String action) {Map<String, String> curStateMap = stateMap.get(curState);if (curStateMap == null) {System.out.println("curState error");return "";}String nextState = curStateMap.get(action);if (nextState == null) {System.out.println("acction error");return "";}return nextState;}public static void main(String[] args) {StateMachine sm = new StateMachine();String nextState = sm.getNextState("init", "runAction");System.out.println(nextState);nextState = sm.getNextState("run", "stopAction");System.out.println(nextState);nextState = sm.getNextState("stop", "runAction");System.out.println(nextState);nextState = sm.getNextState("run", "endAction");System.out.println(nextState);}}

三、总结

        状态机模式是一种强大的工具,能够帮助开发者在软件开发中处理复杂的逻辑和状态管理问题。通过明确定义系统的状态和转换规则,状态机模式使得系统的行为更加清晰、可控和易于维护。同时,状态机模式还具有广泛的应用场景和灵活的实现方式,适用于多种复杂的软件开发需求。

如果状态模式对你有用,记得点赞收藏。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/41202.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

STM32F103C8T6核心板原理图和PCB分享

PCB图 原理图 资料下载地址&#xff1a; 原理图PCB库: https://545c.com/d/45573183-61875742-29897c?p7526 (访问密码: 7526)

[go-zero] 简单微服务调用

文章目录 1.注意事项2.服务划分及创建2.1 用户微服务2.2 订单微服务 3.启动服务3.1 etcd 服务启动3.2 微服务启动3.3 测试访问 1.注意事项 go-zero微服务的注册中心默认使用的是Etcd。 本小节将以一个订单服务调用用户服务来简单演示一下&#xff0c;其实订单服务是api服务&a…

Java 使用sql查询mongodb

在现代应用开发中&#xff0c;关系型数据库和NoSQL数据库各有千秋。MongoDB作为一种流行的NoSQL数据库&#xff0c;以其灵活的文档模型和强大的扩展能力&#xff0c;受到广泛欢迎。然而&#xff0c;有时开发者可能更熟悉SQL查询语法&#xff0c;或者需要在现有系统中复用SQL查询…

【ARMv8/v9 GIC 系列 5.6 -- GIC 超优先级中断详细介绍】

请阅读【ARM GICv3/v4 实战学习 】 文章目录 Interrupt superpriority超优先级中断的特性和应用Physical interface interrupt signalsPhysical Group 1 Non-NMI for Current Security StatePhysical Group 1 for Other Security State, or a Group 0 Non-NMIPhysical Group 1 …

进程控制-wait和waitpid进程回收

wait 阻塞函数 函数作用&#xff1a; 1. 阻塞并等待子进程退出 2. 回收子进程残留资源 3. 获取子进程结束状态&#xff08;退出原因&#xff09; pid_t wait(int *wstatus); 返回值&#xff1a; ‐1 : 回收失败&#xff0c;已经没有子进程了 >0 : 回收子进程对应的…

一种非凸全变差正则化的信号降噪方法(以模拟信号和轴承振动信号为例,MATLAB)

以旋转机械振动信号为例&#xff0c;由于旋转机械运行中背景噪声较强&#xff0c;振动信号需要进行降噪处理。常用的小波阈值降噪会在信号的不连续处产生虚假的波峰和伪吉布森震荡&#xff0c;而奇异值分解SVD去噪容易产生虚假分量&#xff0c;全变差去噪则不会出现这样的情况&…

深入理解JS逆向代理与环境监测

博客文章&#xff1a;深入理解JS逆向代理与环境监测 1. 引言 首先要明确JavaScript&#xff08;JS&#xff09;在真实网页浏览器环境和Node.js环境中有很多使用特性的区别。尤其是在环境监测和对象原型链的检测方面。本文将探讨如何使用JS的代理&#xff08;Proxy&#xff09…

STM32-USART

本内容基于江协科技STM32视频学习之后整理而得。 文章目录 1. 串口通信协议1.1 通信接口1.2 串口通信1.3 硬件电路1.4 电平标准1.5 串口参数及时序1.6 串口时序 2. USART串口通信2.1 USART简介2.2 USART框图2.3 USART基本结构2.4 数据帧2.5 数据帧-配置停止位2.6 起始位侦测2.…

【Python】一文向您详细介绍 argparse中 action=‘store_true’ 的作用

【Python】一文向您详细介绍 argparse中 action‘store_true’ 的作用 下滑即可查看博客内容 &#x1f308; 欢迎莅临我的个人主页 &#x1f448;这里是我静心耕耘深度学习领域、真诚分享知识与智慧的小天地&#xff01;&#x1f387; &#x1f393; 博主简介&#xff1a;98…

pdf怎么转换成图片格式文件,pdf文档怎么转换成图片格式

在数字化时代&#xff0c;pdf文件转换成图片格式是一种常见的操作&#xff0c;无论是在工作还是日常生活中&#xff0c;我们总会遇到需要将pdf文件转换为图片的需求。这可能是因为图片格式更易于分享、展示或编辑。那么&#xff0c;如何高效地将pdf转换成图片呢&#xff1f;本文…

图神经网络实战(16)——经典图生成算法

图神经网络实战&#xff08;16&#xff09;——经典图生成算法 0. 前言1. 图生成技术2. Erdős–Rnyi模型3. 小世界模型小结系列链接 0. 前言 图生成算法是指用于创建模拟图或网络结构的算法&#xff0c;这些算法可以根据特定的规则和概率分布生成具有特定属性的图&#xff0c…

将大型语言模型模块化打造协作智能体

B UILDING C OOPERATIVE E MBODIED A GENTS MODULARLY WITH L ARGE L ANGUAGE M ODELS 论文链接&#xff1a; https://arxiv.org/abs/2307.02485https://arxiv.org/abs/2307.02485 1.概述 在去中心化控制及多任务环境中&#xff0c;多智能体合作问题因原始感官观察、高昂…

【机器学习】机器学习重塑广告营销:精准触达,高效转化的未来之路

&#x1f4dd;个人主页&#x1f339;&#xff1a;Eternity._ &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; ❀目录 &#x1f4d2;1. 引言&#x1f4d9;2. 机器学习基础与广告营销的结合&#x1f9e9;机器学习在广告营销中的核心应用领域&#x1f339;用…

【React】React18 Hooks 之 useReducer

目录 useReducer案例1&#xff1a;useReducer不带初始化函数案例2&#xff1a;useReducer带初始化函数注意事项1&#xff1a;dispatch函数不会改变正在运行的代码的状态注意事项2&#xff1a;获取dispatch函数触发后 JavaScript 变量的值注意事项3&#xff1a;触发了reducer&am…

Spring Boot集成olingo快速入门demo

1.什么是olingo&#xff1f; Apache Olingo 是个 Java 库&#xff0c;用来实现 Open Data Protocol (OData)。 Apache Olingo 包括服务客户端和 OData 服务器方面。 Open Data Protocol &#xff08;开放数据协议&#xff0c;OData&#xff09; 是用来查询和更新数据的一种W…

【吊打面试官系列-MyBatis面试题】MyBatis 实现一对多有几种方式,怎么操作的?

大家好&#xff0c;我是锋哥。今天分享关于 【MyBatis 实现一对多有几种方式,怎么操作的&#xff1f;】面试题&#xff0c;希望对大家有帮助&#xff1b; MyBatis 实现一对多有几种方式,怎么操作的&#xff1f; 有联合查询和嵌套查询。联合查询是几个表联合查询,只查询一次,通过…

观察矩阵(View Matrix)、投影矩阵(Projection Matrix)、视口矩阵(Window Matrix)及VPM矩阵及它们之间的关系

V表示摄像机的观察矩阵&#xff08;View Matrix&#xff09;&#xff0c;它的作用是把对象从世界坐标系变换到摄像机坐标系。因此&#xff0c;对于世界坐标系下的坐标值worldCoord(x0, y0, z0)&#xff0c;如果希望使用观察矩阵VM将其变换为摄像机坐标系下的坐标值localCoord(x…

Node.js-path 模块

path 模块 path 模块提供了 操作路径 的功能&#xff0c;如下是几个较为常用的几个 API&#xff1a; 代码实例&#xff1a; const path require(path);//获取路径分隔符 console.log(path.sep);//拼接绝对路径 console.log(path.resolve(__dirname, test));//解析路径 let pa…

vulhub-activemq(CVE-2016-3088)

在 Apache ActiveMQ 5.12.x~5.13.x 版本中&#xff0c;默认关闭了 fileserver 这个应用&#xff08;不过&#xff0c;可以在conf/jetty.xml 中开启&#xff09;&#xff1b;在 5.14.0 版本后&#xff0c;彻底删除了 fileserver 应用。【所以在渗透测试过程中要确定好 ActiveMQ …

数据结构1:C++实现变长数组

数组作为线性表的一种&#xff0c;具有内存连续这一特点&#xff0c;可以通过下标访问元素&#xff0c;并且下标访问的时间复杂的是O(1)&#xff0c;在数组的末尾插入和删除元素的时间复杂度同样是O(1)&#xff0c;我们使用C实现一个简单的边长数组。 数据结构定义 class Arr…