当前位置: 首页 > news >正文

Java设计模式 十八 状态模式 (State Pattern)

状态模式 (State Pattern)

状态模式是一种行为型设计模式,它允许对象在其内部状态改变时改变其行为。状态模式让一个对象在其状态改变时,其行为也随之改变,看起来就像是改变了对象的类。通过将状态的变化封装到不同的状态对象中,状态模式可以使对象的行为更加灵活和可扩展。


1. 状态模式的组成

状态模式通常包含以下几个角色:

  • Context(上下文):上下文类维护一个当前状态的引用,它通常会委托给具体的状态类来执行相关行为。上下文类还可以动态地切换状态。
  • State(状态接口):定义了状态类的共同接口,所有具体的状态类都实现这个接口。
  • ConcreteState(具体状态类):每个具体的状态类实现了State接口,并定义了在该状态下的具体行为。

2. 状态模式的工作流程

  1. 上下文对象持有一个当前的状态对象,并通过调用状态对象的行为来执行相应的操作。
  2. 每个状态类(ConcreteState)实现了State接口,并提供了在该状态下的具体实现。
  3. 上下文可以在不同状态之间进行切换,通常由外部环境或条件触发。

3. 状态模式的实现

场景示例:电梯系统

我们来实现一个简单的电梯系统。电梯可以处于多个状态,比如:空闲运行停靠。每个状态下电梯的行为不同。通过状态模式,我们可以根据电梯的不同状态执行不同的行为。


1) 定义状态接口

状态接口定义了电梯所支持的操作,如openDoor()closeDoor()等。

// 状态接口
public interface ElevatorState {void openDoor();void closeDoor();void moveUp();void moveDown();
}

2) 定义具体状态类

每个具体的状态类实现了ElevatorState接口,并定义了该状态下电梯的具体行为。

// 空闲状态
public class IdleState implements ElevatorState {private Elevator elevator;public IdleState(Elevator elevator) {this.elevator = elevator;}@Overridepublic void openDoor() {System.out.println("The door is opening.");elevator.setCurrentState(elevator.getOpenState());}@Overridepublic void closeDoor() {System.out.println("The door is already closed.");}@Overridepublic void moveUp() {System.out.println("The elevator is moving up.");elevator.setCurrentState(elevator.getMovingUpState());}@Overridepublic void moveDown() {System.out.println("The elevator is moving down.");elevator.setCurrentState(elevator.getMovingDownState());}
}// 开门状态
public class OpenState implements ElevatorState {private Elevator elevator;public OpenState(Elevator elevator) {this.elevator = elevator;}@Overridepublic void openDoor() {System.out.println("The door is already open.");}@Overridepublic void closeDoor() {System.out.println("Closing the door.");elevator.setCurrentState(elevator.getIdleState());}@Overridepublic void moveUp() {System.out.println("Cannot move while the door is open.");}@Overridepublic void moveDown() {System.out.println("Cannot move while the door is open.");}
}// 上升状态
public class MovingUpState implements ElevatorState {private Elevator elevator;public MovingUpState(Elevator elevator) {this.elevator = elevator;}@Overridepublic void openDoor() {System.out.println("Cannot open the door while the elevator is moving up.");}@Overridepublic void closeDoor() {System.out.println("The door is already closed.");}@Overridepublic void moveUp() {System.out.println("The elevator is already moving up.");}@Overridepublic void moveDown() {System.out.println("Changing direction to move down.");elevator.setCurrentState(elevator.getMovingDownState());}
}// 下降状态
public class MovingDownState implements ElevatorState {private Elevator elevator;public MovingDownState(Elevator elevator) {this.elevator = elevator;}@Overridepublic void openDoor() {System.out.println("Cannot open the door while the elevator is moving down.");}@Overridepublic void closeDoor() {System.out.println("The door is already closed.");}@Overridepublic void moveUp() {System.out.println("Changing direction to move up.");elevator.setCurrentState(elevator.getMovingUpState());}@Overridepublic void moveDown() {System.out.println("The elevator is already moving down.");}
}

3) 定义上下文类

Elevator类是上下文类,负责维护当前的状态,并委托给具体状态类执行操作。

// 电梯类(上下文)
public class Elevator {private ElevatorState idleState;private ElevatorState openState;private ElevatorState movingUpState;private ElevatorState movingDownState;private ElevatorState currentState;public Elevator() {idleState = new IdleState(this);openState = new OpenState(this);movingUpState = new MovingUpState(this);movingDownState = new MovingDownState(this);currentState = idleState;  // 默认初始状态为空闲状态}// 切换状态public void setCurrentState(ElevatorState state) {this.currentState = state;}// 获取当前状态public ElevatorState getCurrentState() {return currentState;}public ElevatorState getIdleState() {return idleState;}public ElevatorState getOpenState() {return openState;}public ElevatorState getMovingUpState() {return movingUpState;}public ElevatorState getMovingDownState() {return movingDownState;}// 电梯操作public void openDoor() {currentState.openDoor();}public void closeDoor() {currentState.closeDoor();}public void moveUp() {currentState.moveUp();}public void moveDown() {currentState.moveDown();}
}

4) 客户端代码

客户端代码模拟了电梯的操作,演示了不同状态下电梯的行为。

public class Client {public static void main(String[] args) {// 创建电梯对象Elevator elevator = new Elevator();// 电梯空闲时elevator.openDoor();  // 输出:The door is opening.elevator.moveUp();    // 输出:The elevator is moving up.// 电梯上升时elevator.moveDown();  // 输出:Changing direction to move down.elevator.openDoor();  // 输出:Cannot open the door while the elevator is moving down.// 电梯下降时elevator.closeDoor(); // 输出:The door is already closed.elevator.moveUp();    // 输出:Changing direction to move up.}
}

运行结果:
The door is opening.
The elevator is moving up.
Changing direction to move down.
Cannot open the door while the elevator is moving down.
The door is already closed.
Changing direction to move up.

4. 状态模式的优点

  1. 封装状态行为: 每个状态类封装了具体的状态行为,使得不同状态的行为不会混合在一起。
  2. 扩展性: 当需要添加新的状态时,可以通过增加新的状态类而不影响现有的代码,符合开闭原则。
  3. 清晰的状态转换: 状态的转换和每个状态的行为都清晰地定义在具体状态类中,使得代码更容易理解和维护。

5. 状态模式的缺点

  1. 类的数量增加: 每个状态都需要定义一个具体的状态类,当状态种类较多时,可能会导致类的数量激增,增加系统复杂度。
  2. 状态之间的相互依赖: 有时状态之间的转换逻辑较为复杂,可能会引发状态类之间的依赖关系,需要小心设计。

6. 状态模式的应用场景

  • 工作流引擎: 当某个任务根据不同的状态执行不同操作时,状态模式非常适用,例如审批流程中的不同状态(待审批、审批中、已通过、已拒绝等)。
  • 有限状态机: 适用于有限的状态集合,如游戏中的玩家状态(例如,待机、攻击、跳跃等)。
  • GUI组件: 例如,按钮、窗体等可以有不同的状态(激活、禁用、隐藏等),这些状态的行为可以通过状态模式来管理。

7. 总结

状态模式通过将每个状态的行为封装在独立的状态对象中,使得对象的行为随状态变化而变化。这种模式可以有效地管理和扩展具有多个状态的对象,特别是在复杂的状态转移和行为执行场景中,它提供了一种灵活且清晰的解决方案。

http://www.lryc.cn/news/526629.html

相关文章:

  • PyTorch张量操作reshape view permute transpose
  • RabbitMQ5-死信队列
  • macOS使用LLVM官方发布的tar.xz来安装Clang编译器
  • 【算法学习】归并排序算法思想的应用—求逆序对数量
  • 一组开源、免费、Metro风格的 WPF UI 控件库
  • Spring Security 应用详解
  • 业务对象和对象的区别
  • 81,【5】BUUCTF WEB [b01lers2020]Life on Mars
  • 华硕笔记本装win10哪个版本好用分析_华硕笔记本装win10专业版图文教程
  • Linux进程 -fork(初识),进程状态和进程优先级
  • 数据从前端传到后端入库过程分析
  • macOS如何进入 Application Support 目录(cd: string not in pwd: Application)
  • 第38周:猫狗识别 (Tensorflow实战第八周)
  • 【2024年华为OD机试】 (A卷,200分)- 计算网络信号、信号强度(JavaScriptJava PythonC/C++)
  • 【go语言】数组和切片
  • 2025美赛MCM数学建模A题:《石头台阶的“记忆”:如何用数学揭开历史的足迹》(全网最全思路+模型)
  • 使用 Docker Compose 一键启动 Redis、MySQL 和 RabbitMQ
  • 新增自定义数据功能|UWA Gears V1.0.7
  • docker 简要笔记
  • 在Ubuntu上使用Apache+MariaDB安装部署Nextcloud并修改默认存储路径
  • 【JavaEE】-- 计算机是如何工作的
  • 政安晨的AI大模型训练实践三:熟悉一下LF训练模型的WebUI
  • 基于微信小程序的网上订餐管理系统
  • 科技快讯 | 理想官宣:正式收费!WeChat 港币钱包拓宽商户网络;百川智能发布深度思考模型Baichuan-M1-preview
  • 【java数据结构】map和set
  • 飞牛NAS安装过程中的docker源问题
  • Linux(Centos 7.6)命令详解:dos2unix
  • Linux MySQL离线安装
  • 声明,这些内容和我无关
  • ISO:摄影中的光线敏感度密码