Java 状态模式 详解

状态模式详解

一、状态模式概述

状态模式(State Pattern)是一种行为型设计模式,它允许一个对象在其内部状态改变时改变它的行为,使对象看起来似乎修改了它的类。

核心特点

  • 状态封装:将每个状态的行为封装到独立的类中
  • 状态转换:对象在不同状态间透明切换
  • 消除条件判断:用多态代替状态条件判断
  • 开闭原则:新增状态无需修改现有代码

二、状态模式的结构

主要角色

  1. Context:上下文,维护当前状态
  2. State:抽象状态,定义状态接口
  3. ConcreteState:具体状态,实现特定状态行为

三、状态模式的实现

1. 基本实现

// 状态接口
public interface State {void handle(Context context);
}// 具体状态A
public class ConcreteStateA implements State {public void handle(Context context) {System.out.println("处理状态A的行为");context.setState(new ConcreteStateB());}
}// 具体状态B
public class ConcreteStateB implements State {public void handle(Context context) {System.out.println("处理状态B的行为");context.setState(new ConcreteStateA());}
}// 上下文类
public class Context {private State state;public Context(State state) {this.state = state;}public void setState(State state) {this.state = state;}public void request() {state.handle(this);}
}// 使用示例
Context context = new Context(new ConcreteStateA());
context.request(); // 执行状态A的行为,并切换到状态B
context.request(); // 执行状态B的行为,并切换回状态A

2. 更复杂的实现(订单状态)

// 订单状态接口
public interface OrderState {void next(Order order);void prev(Order order);void printStatus();
}// 新建状态
public class NewState implements OrderState {public void next(Order order) {order.setState(new PaidState());}public void prev(Order order) {System.out.println("订单已在初始状态");}public void printStatus() {System.out.println("订单状态:新建");}
}// 已支付状态
public class PaidState implements OrderState {public void next(Order order) {order.setState(new ShippedState());}public void prev(Order order) {order.setState(new NewState());}public void printStatus() {System.out.println("订单状态:已支付");}
}// 订单类(上下文)
public class Order {private OrderState state;public Order() {this.state = new NewState();}public void setState(OrderState state) {this.state = state;}public void nextState() {state.next(this);}public void previousState() {state.prev(this);}public void printStatus() {state.printStatus();}
}

四、状态模式的应用场景

1. 电梯状态控制

// 电梯状态接口
public interface ElevatorState {void openDoor();void closeDoor();void run();void stop();
}// 运行状态
public class RunningState implements ElevatorState {public void openDoor() {System.out.println("电梯运行中不能开门");}public void closeDoor() {System.out.println("电梯门已是关闭状态");}public void run() {System.out.println("电梯已在运行");}public void stop() {System.out.println("电梯停止运行");}
}// 停止状态
public class StoppedState implements ElevatorState {public void openDoor() {System.out.println("电梯门打开");}public void closeDoor() {System.out.println("电梯门关闭");}public void run() {System.out.println("电梯开始运行");}public void stop() {System.out.println("电梯已是停止状态");}
}

2. 游戏角色状态

// 游戏角色状态
public interface CharacterState {void attack();void defend();void move();
}// 正常状态
public class NormalState implements CharacterState {public void attack() {System.out.println("造成100%伤害");}public void defend() {System.out.println("受到100%伤害");}public void move() {System.out.println("移动速度100%");}
}// 中毒状态
public class PoisonedState implements CharacterState {public void attack() {System.out.println("造成80%伤害");}public void defend() {System.out.println("受到120%伤害");}public void move() {System.out.println("移动速度60%");}
}

3. TCP连接状态

// TCP状态接口
public interface TCPState {void open();void close();void acknowledge();
}// 已建立连接状态
public class EstablishedState implements TCPState {public void open() {System.out.println("连接已建立,无需再次打开");}public void close() {System.out.println("关闭TCP连接");}public void acknowledge() {System.out.println("发送ACK响应");}
}// 监听状态
public class ListenState implements TCPState {public void open() {System.out.println("建立TCP连接");}public void close() {System.out.println("未建立连接,无需关闭");}public void acknowledge() {System.out.println("未建立连接,不能发送ACK");}
}

五、状态模式的变体

1. 状态表驱动

public class StateMachine {private Map<State, Map<Event, State>> transitions = new HashMap<>();private State currentState;public void addTransition(State source, Event event, State target) {transitions.computeIfAbsent(source, k -> new HashMap<>()).put(event, target);}public void handleEvent(Event event) {currentState = transitions.get(currentState).get(event);}
}

2. 状态模式与策略模式结合

public class Context {private StateStrategy state;public void setState(StateStrategy state) {this.state = state;}public void execute() {state.execute();state = state.nextState();}
}public interface StateStrategy {void execute();StateStrategy nextState();
}

六、状态模式的优缺点

优点

  1. 单一职责:每个状态一个类
  2. 开闭原则:易于新增状态
  3. 简化上下文:消除大量条件判断
  4. 状态转换显式:状态转换逻辑清晰

缺点

  1. 类数量多:状态多时代码量大
  2. 过度设计:简单状态机可能不适用
  3. 状态共享:状态间共享数据较复杂

七、最佳实践

  1. 合理划分状态:避免状态粒度过细
  2. 共享状态对象:无状态的状态对象可共享
  3. 状态转换控制:集中或分散管理转换逻辑
  4. 结合其他模式:如与策略模式结合
  5. 文档化状态图:维护状态转换关系图

八、总结

状态模式是管理对象状态的有效方案,特别适用于:

  • 对象行为随状态改变而改变
  • 需要消除大量状态条件判断
  • 状态转换逻辑复杂
  • 需要清晰的状态管理

在实际开发中,状态模式常见于:

  • 工作流引擎
  • 游戏角色状态
  • 设备控制(如电梯、自动售货机)
  • 网络协议实现
  • 订单生命周期管理

正确使用状态模式可以创建清晰、可维护的状态管理代码,但需要注意避免过度设计简单场景。

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

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

相关文章

Nginx 配置 HTTPS 与 WSS 完整指南

Nginx 配置 HTTPS 与 WSS 完整指南 本教程将手把手教你如何为网站配置 HTTPS 加密访问&#xff0c;并通过反向代理实现安全的 WebSocket&#xff08;WSS&#xff09;通信。以 https://www.zhegepai.cn 域名为例&#xff0c;完整流程约需 30 分钟完成。 一、前置准备 1.1 域名…

双向链表的理解

背景 代码中经常会出现双向链表&#xff0c;对于双向链表的插入和删除有对应的API函数接口&#xff0c;但直观的图表更容易理解&#xff0c;所以本文会对rt-thread内核代码中提供的双向链表的一些API函数操作进行绘图&#xff0c;方便后续随时查看。 代码块 rt-thread中提供…

大文件上传源码,支持单个大文件与多个大文件

大文件上传源码&#xff0c;支持单个大文件与多个大文件 Ⅰ 思路Ⅱ 具体代码前端--单个大文件前端--多个大文件前端接口后端 Ⅰ 思路 具体思路请参考我之前的文章&#xff0c;这里分享的是上传流程与源码 https://blog.csdn.net/sugerfle/article/details/130829022 Ⅱ 具体代码…

Unity中的静态合批使用整理

静态批处理是一种绘制调用批处理方法&#xff0c;它组合不移动的网格以减少绘制调用。它将组合的网格转换为世界空间&#xff0c;并为它们构建一个共享顶点和索引缓冲区。然后&#xff0c;对于可见网格&#xff0c;Unity 会执行一系列简单的绘制调用&#xff0c;每个调用之间几…

【机器学习中的基本术语:特征、样本、训练集、测试集、监督/无监督学习】

机器学习基本术语详解 1. 特征&#xff08;Feature&#xff09; 定义&#xff1a;数据的属性或变量&#xff0c;用于描述样本的某个方面。作用&#xff1a;模型通过学习特征与目标之间的关系进行预测。示例&#xff1a; 预测房价时&#xff0c;特征可以是 面积、地段、房龄。…

C++学习之路:指针基础

目录 指针介绍与基本用法双重指针函数指针空指针与野指针函数参数的指针传递最后 指针一般在C/C语言学习的后期接触&#xff0c;这样就导致指针给新手一种高深莫测、难以掌握的刻板印象。但实际上指针的使用很简单&#xff0c;并且还能够极大的提高程序的灵活性&#xff0c;帮助…

【服务日志链路追踪】

MDCInheritableThreadLocal和spring cloud sleuth 在微服务架构中&#xff0c;日志链路追踪&#xff08;Logback Distributed Tracing&#xff09; 是一个关键需求&#xff0c;主要用于跟踪请求在不同服务间的调用链路&#xff0c;便于排查问题。常见的实现方案有两种&#x…

Kafka+Zookeeper从docker部署到spring boot使用完整教程

文章目录 一、Kafka1.Kafka核心介绍&#xff1a;​核心架构​核心特性​典型应用 2.Kafka对 ZooKeeper 的依赖&#xff1a;3.去 ZooKeeper 的演进之路&#xff1a;注&#xff1a;&#xff08;本文采用ZooKeeper3.8 Kafka2.8.1&#xff09; 二、Zookeeper1.核心架构与特性2.典型…

JUC系列JMM学习之随笔

JUC: JUC 是 Java 并发编程的核心工具包,全称为 Java Util Concurrent,是 java.util.concurrent 包及其子包的简称。它提供了一套强大且高效的并发编程工具,用于简化多线程开发并提高性能。 CPU核心数和线程数的关系:1核处理1线程(同一时间单次) CPU内核结构: 工作内…

The Rust Programming Language 学习 (九)

泛型 每一个编程语言都有高效处理重复概念的工具。在 Rust 中其工具之一就是 泛型&#xff08;generics&#xff09;。泛型是具体类型或其他属性的抽象替代。我们可以表达泛型的属性&#xff0c;比如他们的行为或如何与其他泛型相关联&#xff0c;而不需要在编写和编译代码时知…

蓝桥杯 混乘数字

问题描述 混乘数字的定义如下&#xff1a; 对于一个正整数 n&#xff0c;如果存在正整数 a 和 b&#xff0c;使得&#xff1a; n a b且 a 与 b 的十进制数位中每个数字出现的次数之和&#xff0c;与 n 中对应数字出现的次数相同&#xff0c;则称 n 为混乘数字。 示例 对于…

CExercise04_1位运算符_2 定义一个函数判断给定的正整数是否为2的幂

题目&#xff1a; 给定一个正整数&#xff0c;请定义一个函数判断它是否为2的幂(1, 2, 4, 8, 16, …) 分析&#xff1a; &#xff1a; 代码 #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdbool.h>/* 给定一个正整数&#xff0c;请定义一个函数…

SSL证书不可信的原因有哪些?(国科云)

SSL证书用于在客户端和服务器端之间建立一条加密通道&#xff0c;确保数据在传输过程中的安全性和完整性。然而&#xff0c;在实际应用中&#xff0c;我们有时会遇到SSL证书不可信的情况&#xff0c;严重影响了用户对网站的信任度。那么&#xff0c;SSL证书不可信的原因究竟有哪…

[王阳明代数讲义]琴语言类型系统工程特性

琴语言类型系统工程特性 层展物理学组织实务与艺术与琴生生.物机.械科.技工.业研究.所软凝聚态物理开发工具包社会科学气质砥砺学人生意气场社群成员魅力场与心气微积分社会关系力学 意气实体过程图论信息编码&#xff0c;如来码导引 注意力机制道装Transformer架构的发展标度律…

自抗扰ADRC之二阶线性扩展状态观测器(LESO)推导

1.龙伯格观测器 实际工程应用中&#xff0c;状态变量有时难以使用传感器直接测量&#xff0c;在这种情况下&#xff0c;使用状态观测器估计系统实际状态是非常常见的做法。最出名的状态观测器当属龙伯格博士在1971年发表于TAC的An Introduction to Observer[1]一文中提出的基于…

从头开发一个Flutter插件(二)高德地图定位插件

开发基于高德定位SDK的Flutter插件 在上一篇文章里具体介绍了Flutter插件的具体开发流程&#xff0c;从创建项目到发布。接下来将为Flutter天气项目开发一个基于高德定位SDK的Flutter定位插件。 申请key 首先进入高德地图定位SDK文档内下载定位SDK&#xff0c;并按要求申请A…

分布式锁之redis6

一、分布式锁介绍 之前我们都是使用本地锁&#xff08;synchronize、lock等&#xff09;来避免共享资源并发操作导致数据问题&#xff0c;这种是锁在当前进程内。 那么在集群部署下&#xff0c;对于多个节点&#xff0c;我们要使用分布式锁来避免共享资源并发操作导致数据问题…

ubuntu中使用安卓模拟器

本文这里介绍 使用 android studio Emulator &#xff0c; 当然也有 Anbox (Lightweight)&#xff0c; Waydroid (Best for Full Android Experience), 首先确保自己安装了 android studio &#xff1b; sudo apt update sudo apt install openjdk-11-jdk sudo snap install…

二语习得理论(Second Language Acquisition, SLA)如何学习英语

二语习得理论&#xff08;Second Language Acquisition, SLA&#xff09;是研究学习者如何在成人或青少年阶段学习第二语言&#xff08;L2&#xff09;的理论框架。该理论主要关注语言习得过程中的认知、社会和文化因素&#xff0c;解释了学习者如何从初学者逐渐变得流利并能够…

WinDbg. From A to Z! 笔记(下)

原文链接: WinDbg. From A to Z! 文章目录 使用WinDbg临界区相关命令示例 -- 查看临界区其他有用的命令 WinDbg中的伪寄存器自动伪寄存器 WinDbg中的表达式其他操作默认的表达式计算方式 WinDbg中的重命名调试器命令语言编程控制流命令程序执行 WinDbg 远程调试事件监控WinDbg …