状态模式
在软件开发过程中,我们经常会遇到这样的情况:一个对象的行为会随着其内部状态的改变而发生变化。例如,一个手机在不同状态下(开机、关机、静音等)对相同的操作(如来电)会有不同的反应。传统的解决方案可能会使用大量的条件判断语句(如 if - else 或 switch - case)来处理不同状态下的行为,但这种方式会使代码变得臃肿、难以维护和扩展。状态模式(State Pattern)应运而生,它提供了一种优雅的方式来处理对象状态变化及其对应的行为变化。
状态模式概述
状态模式是一种行为型设计模式,它允许对象在内部状态改变时改变其行为,就好像对象修改了它的类一样。状态模式主要包含以下几个角色:
- 环境类(Context):也称为上下文,它持有一个状态对象的引用,定义了客户端感兴趣的接口,并将与状态相关的操作委托给当前的状态对象来处理。
- 抽象状态类(State):定义了一个接口,用于封装与环境类的一个特定状态相关的行为。
- 具体状态类(ConcreteState):实现抽象状态类中定义的接口,每个具体状态类对应环境类的一个具体状态,实现该状态下的具体行为。
状态模式代码示例
以下是使用 Java 语言实现状态模式的示例代码。以一个简单的电梯控制系统为例,电梯有 “运行”、“停止” 两种状态,并且在不同状态下对 “开门”、“关门” 操作有不同的响应。
// 抽象状态类
abstract class LiftState {protected LiftContext liftContext;public void setLiftContext(LiftContext liftContext) {this.liftContext = liftContext;}public abstract void open();public abstract void close();public abstract void run();public abstract void stop();
}// 运行状态类
class RunningState extends LiftState {@Overridepublic void open() {System.out.println("电梯正在运行,无法开门");}@Overridepublic void close() {System.out.println("电梯正在运行,门已关闭");}@Overridepublic void run() {System.out.println("电梯正在运行");}@Overridepublic void stop() {System.out.println("电梯停止运行");liftContext.setState(liftContext.getStoppedState());}
}// 停止状态类
class StoppedState extends LiftState {@Overridepublic void open() {System.out.println("电梯停止,门打开");}@Overridepublic void close() {System.out.println("电梯停止,门关闭");liftContext.setState(liftContext.getRunningState());}@Overridepublic void run() {System.out.println("电梯开始运行");liftContext.setState(liftContext.getRunningState());}@Overridepublic void stop() {System.out.println("电梯已停止");}
}// 环境类
class LiftContext {private LiftState state;private RunningState runningState = new RunningState();private StoppedState stoppedState = new StoppedState();public LiftContext() {this.state = stoppedState;runningState.setLiftContext(this);stoppedState.setLiftContext(this);}public void setState(LiftState state) {this.state = state;}public RunningState getRunningState() {return runningState;}public StoppedState getStoppedState() {return stoppedState;}public void open() {state.open();}public void close() {state.close();}public void run() {state.run();}public void stop() {state.stop();}
}public class StatePatternDemo {public static void main(String[] args) {LiftContext liftContext = new LiftContext();liftContext.open();liftContext.close();liftContext.run();liftContext.stop();liftContext.open();}
}
在上述代码中,LiftState 是抽象状态类,定义了电梯在不同状态下可执行的操作接口。RunningState 和 StoppedState 是具体状态类,分别实现了电梯在 “运行” 和 “停止” 状态下的具体行为。LiftContext 是环境类,持有当前电梯的状态,并将对电梯的操作委托给当前状态对象。在 main 方法中,我们模拟了电梯的一系列操作,展示了状态模式下电梯在不同状态下的行为变化。
状态模式的应用场景
- 游戏开发:游戏角色在不同状态下(如站立、奔跑、跳跃、受伤等)对用户输入(如按键操作)会有不同的反应。通过状态模式,可以将不同状态下的行为封装在各自的具体状态类中,使代码结构更加清晰,易于维护和扩展。
- 工作流系统:在工作流系统中,任务可能处于不同的状态(如待处理、处理中、已完成等),并且在不同状态下对各种操作(如分配任务、提交任务等)有不同的处理方式。状态模式可以很好地处理这种情况,将工作流的状态管理和行为处理分离。
- 设备状态管理:对于各种设备(如打印机、空调等),它们在不同状态下(如开启、关闭、故障等)对用户操作(如打印指令、温度调节等)的响应不同。使用状态模式可以方便地实现设备状态的管理和相应行为的处理。
状态模式的优缺点
- 优点
- 清晰的状态管理:状态模式将对象的状态和行为封装在不同的具体状态类中,使得代码结构更加清晰,易于理解和维护。每个状态类专注于实现该状态下的特定行为,符合单一职责原则。
- 易于扩展:当需要添加新的状态或修改现有状态的行为时,只需要创建新的具体状态类或修改现有状态类的代码,而不需要在大量的条件判断语句中进行修改,符合开闭原则。
- 提高可维护性:避免了使用大量的条件判断语句,使得代码更加简洁,减少了出错的可能性。同时,当状态变化时,只需要修改相应状态类的代码,而不会影响其他状态的行为。
- 缺点
- 增加类的数量:由于每个状态都需要一个具体的状态类来实现,可能会导致类的数量增加,使项目的代码结构变得复杂。在小型项目中,这种复杂性可能会显得过于冗余。
- 状态转换逻辑复杂:在一些复杂的应用场景中,状态之间的转换逻辑可能会变得复杂,需要仔细设计和维护状态转换的条件和顺序,否则可能会导致状态转换错误或不符合预期的行为。
结语
希望本文能帮助您更好地理解状态模式的概念及其实际应用。如果您有任何疑问或建议,请随时留言交流。
相关文章:
状态模式
在软件开发过程中,我们经常会遇到这样的情况:一个对象的行为会随着其内部状态的改变而发生变化。例如,一个手机在不同状态下(开机、关机、静音等)对相同的操作(如来电)会有不同的反应。传统的解…...
RoHS 简介
RoHS(Restriction of Hazardous Substances Directive,限制有害物质指令)是欧盟制定的一项环保法规,旨在限制电气和电子设备中某些有害物质的使用,以减少这些产品对环境和人体健康的危害。 RoHS限制的有害物质及其限量…...
【Vim Masterclass 笔记26】S11L46:Vim 插件的安装、使用与日常管理
文章目录 Section 11:Vim PluginsS11L46 Managing Vim Plugins1 第三方插件管理工具2 安装插件使用的搜索引擎3 Vim 插件的安装方法4 存放 Vim 插件包的路径格式5 示例一:插件 NERDTree 的安装6 示例二:插件 ctrlp.vim 的安装7 示例三&#x…...
深度学习原理与Pytorch实战
深度学习原理与Pytorch实战 第2版 强化学习人工智能神经网络书籍 python动手学深度学习框架书 TransformerBERT图神经网络: 技术讲解 编辑推荐 1.基于PyTorch新版本,涵盖深度学习基础知识和前沿技术,由浅入深,通俗易懂…...
ELK环境搭建
文章目录 1.ElasticSearch安装1.安装的版本选择1.SpringBoot版本:2.4.2 找到依赖的spring-data-elasticsearch的版本2.spring-data-elasticsearch版本:4.1.3 找到依赖的elasticsearch版本3.elasticsearch版本:7.9.3 2.安装1.官方文档2.下载压…...
基于Springboot + vue实现的民俗网
“前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站:人工智能学习网站” 💖学习知识需费心, 📕整理归纳更费神。 🎉源码免费人人喜…...
第24篇 基于ARM A9处理器用汇编语言实现中断<六>
Q:怎样设计ARM处理器汇编语言程序使用定时器中断实现实时时钟? A:此前我们曾使用轮询定时器I/O的方式实现实时时钟,而在本实验中将采用定时器中断的方式。新增第三个中断源A9 Private Timer,对该定时器进行配置&#…...
【数据结构】_不带头非循环单向链表
目录 1. 链表的概念及结构 2. 链表的分类 3. 单链表的实现 3.1 SList.h头文件 3.2 SList.c源文件 3.3 Test_SList.c测试文件 关于线性表,已介绍顺序表,详见下文: 【数据结构】_顺序表-CSDN博客 本文介绍链表; 基于顺序表…...
golang 使用双向链表作为container/heap的载体
MyHeap:container/heap的数据载体,需要实现以下方法: Len:堆中数据个数 Less:第i个元素 是否必 第j个元素 值小 Swap:交换第i个元素和 第j个元素 Push:向堆中追加元素 Pop:从堆…...
C#集合操作优化:高效实现批量添加与删除
在C#中,对集合进行批量操作(如批量添加或删除元素)通常涉及使用集合类型提供的方法和特性,以及可能的循环或LINQ查询来高效地处理大量数据。以下是一些常见的方法和技巧: 批量添加元素 使用集合的AddRange方法&#x…...
142.WEB渗透测试-信息收集-小程序、app(13)
免责声明:内容仅供学习参考,请合法利用知识,禁止进行违法犯罪活动! 内容参考于: 易锦网校会员专享课 上一个内容:141.WEB渗透测试-信息收集-小程序、app(12) 软件用法,…...
24.日常算法
1. 数组中两元素的最大乘积 题目来源 给你一个整数数组 nums,请你选择数组的两个不同下标 i 和 j,使 (nums[i]-1)*(nums[j]-1) 取得最大值。请你计算并返回该式的最大值。 示例 1: 输入:nums [3,4,5,2] 输出:12 解释…...
分布式理解
分布式 如何理解分布式 狭义的分布是指,指多台PC在地理位置上分布在不同的地方。 分布式系统 分布式系**统:**多个能独立运行的计算机(称为结点)组成。各个结点利用计算机网络进行信息传递,从而实现共同的“目标或者任…...
wordpress调用指定ID页面的链接
在WordPress中,如果你想调用一个指定ID的页面链接,可以使用以下几种方法: 方法一:使用页面ID 你可以直接使用页面的ID来生成链接。例如,如果你想链接到ID为123的页面,可以使用以下代码: <…...
单值二叉树(C语言详解版)
一、摘要 今天要讲的是leetcode单值二叉树,这里用到的C语言,主要提供的是思路,大家看了我的思路之后可以点击链接自己试一下。 二、题目简介 如果二叉树每个节点都具有相同的值,那么该二叉树就是单值二叉树。 只有给定的树是单…...
python学opencv|读取图像(四十二)使用cv2.add()函数实现多图像叠加
【1】引言 前序学习过程中,掌握了灰度图像和彩色图像的掩模操作: python学opencv|读取图像(九)用numpy创建黑白相间灰度图_numpy生成全黑图片-CSDN博客 python学opencv|读取图像(四十)掩模:三…...
速通Docker === Docker Compose
目录 Docker Compose 简介 Docker Compose 常用命令 使用 Docker Compose 启动 WordPress 普通启动方式(使用 Docker 命令) 使用 Docker Compose 启动 Docker Compose 的特性 Docker Compose 简介 Docker Compose 是一个用于定义和运行多容器 Dock…...
LMI Gocator GO_SDK VS2019引用配置
LMI SDK在VS2019中的引用是真的坑爹,总结一下经验,希望后来的人能少走弯路.大致内容如下: (1) 环境变量 (2)C/C 附加包含目录 E:\GWQ\Gocator\GO_SDK\Gocator\GoSdk E:\GWQ\Gocator\GO_SDK\Platform\kApi (3&#…...
技术之翼,创作之心
引言:初入编程的迷茫与追求 当我第一次接触到编程时,心中充满了既期待又迷茫的情感。那时,我还是一名刚刚踏入大学的学生,面对一门陌生而复杂的学科——计算机科学,我的内心充满了好奇与困惑。课堂上,老师…...
WebSocket异步导出
WebSocket异步导出 1、安装sockjs-client和stompjs2、连接后台3、vite.config.ts 配置反向代理4、导出并实时通信5、 封装WebSocket 文件注册登录(城通网盘) 1、安装sockjs-client和stompjs import SockJS from sockjs-client/dist/sockjs.min.js import Stomp from stompjs2、…...
Obsidian PDF++:如何在Obsidian中实现PDF与笔记的无缝双向链接?
Obsidian PDF:如何在Obsidian中实现PDF与笔记的无缝双向链接? 【免费下载链接】obsidian-pdf-plus PDF: the most Obsidian-native PDF annotation & viewing tool ever. Comes with optional Vim keybindings. 项目地址: https://gitcode.com/gh_…...
深圳实体门店有必要做GEO AI代运营吗
深圳实体门店有必要做GEO AI代运营吗一、开篇引言2026年深圳本地实体商业竞争进入白热化阶段,全城数百万家线下实体门店涵盖本地生活、家装工装、汽车服务、餐饮娱乐、教育培训等全品类,传统线下地推、门店自然客流、传统团购平台引流效果持续下滑&#…...
DeepSeek系统设计辅助:如何在48小时内完成可审计、可回滚、可压测的AI服务架构图?
更多请点击: https://intelliparadigm.com 第一章:DeepSeek系统设计辅助 DeepSeek系统设计辅助模块面向架构师与后端工程师,提供模型能力调用、接口契约生成、异步任务编排等核心支撑能力。该模块不替代人工设计决策,而是通过结构…...
上线前最后一道防线,DeepSeek代码审查如何帮你拦截87%的CVE类缺陷?
更多请点击: https://intelliparadigm.com 第一章:上线前最后一道防线,DeepSeek代码审查如何帮你拦截87%的CVE类缺陷? 在软件交付生命周期末期,传统人工代码审计与通用SAST工具常因误报率高、上下文理解弱而漏检高危漏…...
基于ESP32的智能电池充电器设计:多化学体系支持与模块化架构
1. 项目概述:打造一台全能的“电池医生”手头攒了一堆不同化学体系的电池,从航模用的4S锂聚合物电池,到应急灯里的12V铅酸电池,再到各种工具里的镍氢、锂离子电池,每次充电都得翻出好几个不同的充电器,桌面…...
当 AI Coding 进入复杂企业系统,为什么提效远没有宣传里那么美好 ?
以 Claude Code、Codex 为代表的自主编码智能体(Coding Agents),正在以惊人的速度席卷软件开发者生态。与此同时,类似“10 倍开发效率”“普通人也能随手构建软件”“程序员即将失业”的说法,也随处可见。这种不分场景…...
ShrinkBox后门攻击:如何让自动驾驶模型“看错”距离,威胁ML-ADAS安全
1. 项目概述在自动驾驶和高级驾驶辅助系统(ADAS)领域,基于机器学习的目标检测模型,如YOLO系列,已成为感知环境、实现碰撞预警的核心组件。这些模型通过实时识别和定位道路上的车辆、行人等目标,为后续的距离…...
用Azure Kinect DK和Body Tracking SDK,5分钟实现一个实时人体骨骼点检测Demo(C++版)
5分钟实战:用Azure Kinect DK实现实时人体骨骼点追踪(C版) 当你第一次拿到Azure Kinect DK时,最令人兴奋的莫过于它强大的人体追踪能力。这款深度相机不仅能捕捉高清彩色图像,更能通过AI算法实时重建人体骨骼关节点。本…...
别再只用鼠标了!用Leap Motion手势控制Unity游戏,保姆级配置避坑指南(2024版)
2024年Unity手势交互开发实战:Leap Motion从配置到游戏逻辑全解析在游戏开发领域,交互方式的创新往往能带来全新的体验。想象一下,玩家不再需要键盘鼠标,仅凭自然的手部动作就能操控游戏角色——这正是Leap Motion手势识别技术为U…...
Java项目中如何提升整体系统性能?
性能优化可以说是我们程序员的必修课,如果你想要跳出CRUD的苦海,成为一个更“高级”的程序员的话,性能优化这一关你是无论无何都要去面对的。为了提升系统性能,开发人员可以从系统的各个角度和层次对系统进行优化。除了最常见的代…...
