设计模式 行为型 状态模式(State Pattern)与 常见技术框架应用 解析

状态模式(State Pattern)是一种行为型设计模式,它允许对象在内部状态改变时改变其行为,使得对象看起来好像修改了它的类。这种设计模式的核心思想是将对象的状态和行为封装成不同的状态类,通过状态对象的行为改变来避免大量的条件判断语句,从而提升代码的可扩展性和可维护性。
一、核心思想
状态模式的核心思想是将状态相关的行为抽取到独立的状态类中,使得增加新状态变得简单,且不影响其他状态。每个状态类都封装了对象在该状态下的行为,上下文类(Context)则维护一个状态对象,并在状态发生变化时改变其行为。
二、定义与结构
定义:
状态模式(State Pattern)是行为设计模式的一种,它允许一个对象在其内部状态改变时改变它的行为。
结构:
状态模式通常涉及以下几个关键组件:
- State(状态):定义一个接口或抽象类,用于封装与上下文相关的行为。
- ConcreteState(具体状态):实现状态接口或继承抽象状态类,具体定义每个状态下的行为。
- Context(上下文):维护一个状态对象,并在状态发生变化时改变其行为。
三、角色
- 环境角色(Context):也称为上下文,它定义了客户程序与状态对象交互的接口,并且保存了一个具体状态对象的引用。
- 抽象状态(State):这是一个接口或抽象类,定义了所有具体状态所共有的一些行为。
- 具体状态(Concrete States):这些是实现了抽象状态接口的具体类,每个类代表了一个状态,并且在该状态下定义了一些行为。
四、实现步骤及代码示例
以Java为例,状态模式的实现步骤通常包括:
- 定义状态接口:定义一个接口,用于封装与上下文相关的行为。
- 创建具体状态类:实现状态接口,具体定义每个状态下的行为。
- 定义上下文类:维护一个状态对象,并提供方法来切换状态和处理请求。
以下是一个简单的Java代码示例,用于展示状态模式的实现:
// 状态接口
interface State {void handle();
}// 具体状态类:待支付状态
class PendingState implements State {@Overridepublic void handle() {System.out.println("订单状态:待支付。请完成支付!");}
}// 具体状态类:已支付状态
class PaidState implements State {@Overridepublic void handle() {System.out.println("订单状态:已支付。订单处理中。");}
}// 具体状态类:已完成状态
class CompletedState implements State {@Overridepublic void handle() {System.out.println("订单状态:已完成。");}
}// 上下文类
class OrderContext {private State state;public OrderContext() {this.state = new PendingState(); // 初始状态为待支付}public void setState(State state) {this.state = state;}public void handleOrder() {state.handle();}
}// 客户端代码
public class Client {public static void main(String[] args) {OrderContext orderContext = new OrderContext();orderContext.handleOrder(); // 输出:订单状态:待支付。请完成支付!orderContext.setState(new PaidState());orderContext.handleOrder(); // 输出:订单状态:已支付。订单处理中。orderContext.setState(new CompletedState());orderContext.handleOrder(); // 输出:订单状态:已完成。}
}
五、常见技术框架应用
1、前端框架中状态模式的应用
状态模式(State Pattern)是一种行为型设计模式,它允许对象在内部状态发生改变时改变其行为。在前端框架中,状态模式的应用十分广泛,特别是在管理复杂的用户交互逻辑和页面状态时。以下是一些前端框架中应用状态模式的举例:
购物车状态管理
在电商网站的购物车页面中,商品的状态(如已添加、已删除、数量变化等)会影响购物车的整体行为和显示。使用状态模式,可以清晰地管理这些状态变化。
- 状态定义:定义购物车状态的抽象类,如
CartState,并为其定义添加商品、删除商品等抽象方法。 - 具体状态类:创建具体的状态类,如
EmptyCartState(空购物车状态)和NonEmptyCartState(非空购物车状态),在这些类中实现具体的行为。 - 购物车类:在购物车类中,维护一个当前状态的引用,并根据状态的变化调用相应的方法。
例如,在购物车为空时,用户只能添加商品;在购物车非空时,用户可以查看商品列表、删除商品等。通过状态模式,可以轻松地管理这些状态变化,使代码更加清晰和可维护。
表单验证状态管理
在用户提交表单之前,需要对表单中的输入进行验证。使用状态模式,可以根据不同的验证状态(如有效、无效、必填项未填写等)来改变表单的行为。
- 状态定义:定义表单验证状态的抽象类,如
FormValidationState,并为其定义验证方法。 - 具体状态类:创建具体的状态类,如
ValidState(有效状态)、InvalidState(无效状态)和RequiredFieldMissingState(必填项未填写状态),在这些类中实现具体的验证逻辑。 - 表单类:在表单类中,维护一个当前验证状态的引用,并根据状态的变化调用相应的方法来处理验证结果。
游戏状态管理
在游戏中,对象的状态可能会随着游戏的进程而发生变化,如角色的生命值、能量值、游戏阶段等。使用状态模式,可以轻松地管理这些状态变化。
- 状态定义:定义游戏状态的抽象类,如
GameState,并为其定义更新、渲染等方法。 - 具体状态类:创建具体的状态类,如
IdleState(空闲状态)、PlayingState(游戏进行状态)、GameOverState(游戏结束状态),在这些类中实现具体的游戏逻辑。 - 游戏管理类:在游戏管理类中,维护一个当前游戏状态的引用,并根据状态的变化调用相应的方法来处理游戏逻辑。
UI组件状态管理
在前端开发中,一些UI组件(如按钮、输入框、下拉菜单等)的状态可能会随着用户的交互而发生变化。使用状态模式,可以清晰地管理这些状态变化。
- 状态定义:定义UI组件状态的抽象类,如
ComponentState,并为其定义更新、渲染等方法。 - 具体状态类:创建具体的状态类,如
HoverState(悬停状态)、ActiveState(激活状态)、DisabledState(禁用状态),在这些类中实现具体的UI逻辑。 - 组件类:在组件类中,维护一个当前状态的引用,并根据状态的变化调用相应的方法来处理UI逻辑。
前端路由状态管理
在单页面应用(SPA)中,前端路由的状态管理是一个重要的问题。使用状态模式,可以清晰地管理路由的状态变化。
- 状态定义:定义路由状态的抽象类,如
RouteState,并为其定义导航、回退等方法。 - 具体状态类:创建具体的状态类,如
HomeState(首页状态)、DetailState(详情页状态)等,在这些类中实现具体的导航逻辑。 - 路由管理类:在路由管理类中,维护一个当前路由状态的引用,并根据状态的变化调用相应的方法来处理导航逻辑。
综上所述,状态模式在前端框架中的应用十分广泛,可以用于管理复杂的用户交互逻辑、页面状态、游戏状态以及UI组件状态等。通过状态模式,可以使代码更加清晰、可维护和可扩展。
2、 Android 中的 View 状态
- 在Android开发中,View(如Button)有多种状态,比如“正常状态”、“按下状态”、“不可用状态”等。这些状态的行为和外观是不同的。
// 定义抽象状态接口(简化版)
interface ViewState {void onDraw(Canvas canvas);boolean onTouchEvent(MotionEvent event);
}
// 正常状态类
class NormalState implements ViewState {@Overridepublic void onDraw(Canvas canvas) {// 绘制正常状态下的视图外观}@Overridepublic boolean onTouchEvent(MotionEvent event) {if (event.getAction() == MotionEvent.ACTION_DOWN) {// 切换到按下状态view.setState(view.getPressedState());return true;}return false;}
}
// 按下状态类
class PressedState implements ViewState {@Overridepublic void onDraw(Canvas canvas) {// 绘制按下状态下的视图外观}@Overridepublic boolean onTouchEvent(MotionEvent event) {if (event.getAction() == MotionEvent.ACTION_UP) {// 切换到正常状态或者执行点击操作等view.setState(view.getNormalState());return true;}return false;}
}
// 自定义View(环境类)
class CustomView extends View {private ViewState state;private ViewState normalState;private ViewState pressedState;public CustomView(Context context) {super(context);normalState = new NormalState();pressedState = new PressedState();state = normalState;}@Overrideprotected void onDraw(Canvas canvas) {state.onDraw(canvas);}@Overridepublic boolean onTouchEvent(MotionEvent event) {return state.onTouchEvent(event);}public void setState(ViewState state) {this.state = state;invalidate();}public ViewState getState() {return state;}public ViewState getNormalState() {return normalState;}public ViewState getPressedState() {return pressedState;}
}
六、应用场景
状态模式适用于以下场景:
- 必须在对象运行时改变对象的行为。
- 传统编程需要考虑所有可能发生的情况,使用条件选择语句if-else判断并选择执行。当新增状态时,需要新增if-else语句,程序扩展繁琐。针对条件表达式过于复杂,可以采用状态模式,分离判断逻辑变为一系列的状态类,使判断逻辑变得更加简单。
七、优缺点
优点:
- 封装性强:每个状态都被封装成一个类,使得状态逻辑更加清晰,便于维护和扩展。
- 避免条件语句:通过状态模式可以避免使用大量的条件语句来控制对象的行为,使代码更加简洁和可读。
- 增强可扩展性:可以轻松地添加新的状态类,而不需要修改已有的代码,从而增强了系统的可扩展性。
缺点:
- 状态类过多可能导致复杂性增加:如果系统中存在大量的状态类,可能会导致状态类之间的关系变得复杂,增加系统的理解和维护成本。
- 状态切换可能不够灵活:在某些情况下,状态切换可能受到限制,无法满足特定的业务需求。
综上所述,状态模式是一种非常有用的设计模式,可以有效地管理对象的状态和行为,并实现状态之间的转换逻辑。在实际应用中,应根据具体业务需求来设计状态类和上下文类,以确保模式的正确应用和系统的稳定性。

相关文章:
设计模式 行为型 状态模式(State Pattern)与 常见技术框架应用 解析
状态模式(State Pattern)是一种行为型设计模式,它允许对象在内部状态改变时改变其行为,使得对象看起来好像修改了它的类。这种设计模式的核心思想是将对象的状态和行为封装成不同的状态类,通过状态对象的行为改变来避免…...
计算机网络 (38)TCP的拥塞控制
前言 TCP拥塞控制是传输控制协议(Transmission Control Protocol,TCP)避免网络拥塞的算法,是互联网上主要的一个拥塞控制措施。 一、目的 TCP拥塞控制的主要目的是防止过多的数据注入到网络中,使网络能够承受现有的网络…...
鸿蒙面试 2025-01-09
鸿蒙分布式理念?(个人认为理解就好) 鸿蒙操作系统的分布式理念主要体现在其独特的“流转”能力和相关的分布式操作上。在鸿蒙系统中,“流转”是指涉多端的分布式操作,它打破了设备之间的界限,实现了多设备…...
【关于for循环的几种写法】
关于for循环的几种写法 在 C 中,for(int i 0; i < n; i) 是一种常见的循环写法,用于遍历从 0 到 n-1 的索引。如果你希望简化这种写法,可以使用以下几种方法: 1. 使用范围 for 循环 如果你需要遍历一个容器(如数…...
Apache和PHP:构建动态网站的黄金组合
在当今的互联网世界,网站已经成为了企业、个人和机构展示自己、与用户互动的重要平台。而在这些动态网站的背后,Apache和PHP无疑是最受开发者青睐的技术组合之一。这一组合提供了高效、灵活且可扩展的解决方案,帮助您快速搭建出强大的网站&am…...
免费开源的下载工具Xdown
软件介绍 Xdown是一款功能强大的开源免费下载工具,专为PC端用户设计,支持多种协议和下载方式。 1、多线程下载 Xdown支持最高128线程的并发下载,能够将文件分割成多个部分同时下载,从而显著提升下载速度。 2、多种协议支持 该…...
Three.js 数学工具:构建精确3D世界的基石
文章目录 前言一、向量(Vectors)二、矩阵(Matrices)三、四元数(Quaternions)四、欧拉角(Euler Angles)五、颜色(Colors)六、几何体生成器(Geometr…...
如何明智地提问
如何明智地提问的重要总结,让我为主要观点添加一些具体的实践建议: 提问前的准备工作 尝试在 Google、Stack Overflow 等平台搜索相似问题阅读相关文档和错误日志尝试自己调试和排查问题记录下已尝试过的解决方案 选择合适的提问平台 Stack Overflow…...
Microsoft Sql Server 2019 函数理解
说到函数,首先和存储过程作个比较吧,两者有一个共同点都是预编译优化后存储在磁盘中,所以效率 要比T-SQL高一点点。值得注意的是,存储过程可以创建或访问临时表,而函数不可以; 同时函数不可 以修改表中的数…...
自定义日期转换配置
文章目录 1.日期问题出现原因以及解决方案概述1.图示2.三种解决方案概述1.对于表单数据 application/x-www-form-urlencoded2.对于JSON数据1.使用JsonFormat注解2.自定义Jackson日期转换配置 2.解决方案common-web-starter1.目录2.BaseController.java 使用InitBinder解决表单数…...
“AI智能服务平台系统,让生活更便捷、更智能
大家好,我是资深产品经理老王,今天咱们来聊聊一个让生活变得越来越方便的高科技产品——AI智能服务平台系统。这个系统可是现代服务业的一颗璀璨明珠,它究竟有哪些魅力呢?下面我就跟大家伙儿闲聊一下。 一、什么是AI智能服务平台系…...
SQL美化器优化
文章目录 1.目录2.代码 1.目录 2.代码 package com.sunxiansheng.mybatis.plus.inteceptor;import org.apache.ibatis.executor.statement.StatementHandler; import org.apache.ibatis.mapping.*; import org.apache.ibatis.plugin.*; import org.apache.ibatis.reflection.*…...
我的128天创作之路:回顾与展望
大家好呀!今天来和你们分享一下我的创作历程😁。 一、机缘 最开始创作呢,是因为在学习 C 的 STL 时,像 string、list、vector 这些模板可把我折腾得够呛,但也让我学到了超多东西!我就想,要是把我…...
内核配置参数整理
#参考网页 linux5.2 <.config>文件注释 详细解释 CONFIG_ARMy:启用ARM架构支持,这是ARM处理器专用的内核配置选项。 CONFIG_ARM_HAS_SG_CHAINy:启用对散列表(scatter-gather)链的支持…...
SpringBoot整合Easy-es
一.什么是Easy-Es Easy-Es(简称EE)是一款基于ElasticSearch(简称Es)官方提供的RestHighLevelClient打造的ORM开发框架,在 RestHighLevelClient 的基础上,只做增强不做改变,为简化开发、提高效率而生,您如果有用过Mybatis-Plus(简称…...
于交错的路径间:分支结构与逻辑判断的思维协奏
大家好啊,我是小象٩(๑ω๑)۶ 我的博客:Xiao Xiangζั͡ޓއއ 很高兴见到大家,希望能够和大家一起交流学习,共同进步。* 这一节内容很多,文章字数达到了史无前例的一万一,我们要来学习分支与循环结构中…...
Linux之读者写者模型与特殊锁的学习
目录 读者写者模型 特殊锁 悲观锁 自旋锁 在前几期,我们学习了多线程的生产者和消费者模型,生产者和消费者模型中,有三种关系,两个角色,一个场所,那么读者写者模型和生产者消费者模型有什么关联吗&…...
回溯专题 记录
回溯的题目按照这套模板进行; 我感觉整体逻辑还是递归,只不过有了pop_back才是回溯概念; class Solution {public:vector<int> path;vector<vector<int>> ans;void backtracking(int n,int k,int startindex){if(path.…...
使用 Python 实现自动化办公(邮件、Excel)
目录 一、Python 自动化办公的准备工作 1.1 安装必要的库 1.2 设置邮件服务 二、邮件自动化处理 2.1 发送邮件 示例代码 注意事项 2.2 接收和读取邮件 示例代码 三、Excel 自动化处理 3.1 读取和写入 Excel 文件 示例代码 3.2 数据处理和分析 示例代码 四、综合…...
贪心算法笔记
贪心算法笔记 大概内容 贪心就是对于一个问题有很多个步骤,我们在每一个步骤中都选取最优的那一个,最后得出答案。就是在一些函数中可行,但是有些比如二次函数,因为它的转折点不一定最优,就是不可行的。那么如何判断贪心呢?有这么几种 看时间复杂度,一般的就是 O ( n…...
工业AI全流程定制开发:以服务适配需求,做实企业数智化改造
当前工业数智化改造已成为企业提升核心竞争力的关键,但行业内普遍存在一个核心痛点:服务与企业实际需求脱节。不少企业在推进数智化过程中,陷入“重产品、轻适配”的误区,盲目采用标准化AI产品,忽视自身生产流程、设备…...
<数据集>yolo骑行者识别<目标检测>
数据集下载链接https://blog.csdn.net/qq_53332949/article/details/159770308?spm1011.2415.3001.5331数据集格式:VOCYOLO格式 图片数量:13674张 标注数量(xml文件个数):13674 标注数量(txt文件个数):13674 标注类别数&…...
06_Cursor之上下文管理与代码库理解
关键字:上下文管理, 代码库理解, 符号引用, Git集成, 图像上下文, Cursor 06_Cursor之上下文管理与代码库理解 Cursor知识体系 Cursor知识体系(续) | -- 上下文管理层 | -- 代码库级理解 | | -- 项目结构分析 | | -- 依赖关系追…...
Slim模板在微服务架构中的终极应用指南:分布式系统模板管理最佳实践
Slim模板在微服务架构中的终极应用指南:分布式系统模板管理最佳实践 【免费下载链接】slim Slim is a template language whose goal is to reduce the syntax to the essential parts without becoming cryptic. 项目地址: https://gitcode.com/gh_mirrors/sli/s…...
MCP Agent Graph: 基于上下文工程的多智能体系统构建指南
1. 引言: 从单一模型到多智能体协作 1.1 大语言模型的能力边界 大语言模型(LLM)的发展经历了从简单文本生成到复杂推理的演进过程。早期的应用场景主要集中在问答、翻译、摘要等相对独立的任务上,模型作为一个无状态的推理引擎,接收输入并产生输出。然…...
新一代高端工业 HMI 如何重塑现场交互体验?
繁易 FPADX 系列电容触摸屏支持 3D 可视化、多点触控、Web 远程访问与大型工程承载,帮助工业设备实现更高效、更直观、更智能的人机交互体验。在工业自动化持续升级的今天,触摸屏早已不再只是设备上的一个操作界面。对于设备制造商、系统集成商和终端工厂…...
OpenClaw技能开发入门:为千问3.5-9B编写自定义文件处理器
OpenClaw技能开发入门:为千问3.5-9B编写自定义文件处理器 1. 为什么需要自定义文件处理器 上周我在整理项目文档时,发现一个重复性痛点:每次收到同事发来的Markdown文件,都需要手动执行"格式校验→重命名→按日期归档→生成…...
FC-CLIP实战:为什么说“卷积不死”?在开放词汇分割中冻结CLIP主干的深度解析与避坑指南
FC-CLIP技术解析:卷积架构在开放词汇分割中的不可替代性 当整个计算机视觉领域似乎都被Transformer架构席卷时,FC-CLIP论文却掷地有声地宣告"卷积不死"。这个看似反潮流的结论背后,隐藏着哪些被忽视的视觉归纳偏置?冻结…...
Phantom Stealer 凭证窃取机制分析与防御体系研究
摘要 Phantom Stealer 作为 2025 年下半年出现的新型多功能信息窃取木马,以多阶段感染、无文件驻留、强反检测与全维度凭证窃取为核心特征,通过伪装合法软件、脚本混淆、进程注入、 Heaven’s Gate 技术规避等手段,精准窃取浏览器密码、Cooki…...
LeetCode 热题100——11.盛最多水的容器
题目: 给定一个长度为 n 的整数数组 height 。有 n 条垂线,第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。 返回容器可以储存的最大水量。 说明:你不…...
