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

状态模式详解与真实场景案例(Java实现)

模式定义

状态模式(State Pattern) 允许对象在其内部状态改变时改变它的行为,使对象看起来像是修改了它的类。属于行为型设计模式,核心思想是将状态抽象为独立对象,不同状态下行为封装在不同状态类中。

解决的问题

消除庞大的条件分支

避免对象在不同状态下使用大量if-else或switch-case判断状态

状态转换逻辑清晰化

将状态转移规则封装在状态对象中

提高扩展性

新增状态时只需添加新状态类,无需修改现有代码

真实场景案例:电商订单状态流转

需求背景

电商订单存在多种状态:
待支付 → 已支付 → 已发货 → 已收货 → 已完成
每个状态下可执行的操作不同:
待支付状态:允许支付、取消订单
已支付状态:允许发货、退款
已发货状态:允许确认收货
已收货状态:允许完成订单
已完成/已取消状态:不可再操作

代码实现

1. 定义状态接口

// 订单状态接口
public interface OrderState {void pay(OrderContext context);      // 支付void cancel(OrderContext context);   // 取消void ship(OrderContext context);     // 发货void receive(OrderContext context);  // 收货void complete(OrderContext context); // 完成
}

2. 实现具体状态类

待支付状态

public class UnpaidState implements OrderState {@Overridepublic void pay(OrderContext context) {System.out.println("支付成功");context.setState(new PaidState());}@Overridepublic void cancel(OrderContext context) {System.out.println("订单已取消");context.setState(new CancelledState());}// 其他操作不支持@Overridepublic void ship(OrderContext context) {throw new UnsupportedOperationException("待支付状态不能发货");}@Overridepublic void receive(OrderContext context) {throw new UnsupportedOperationException("非法操作");}@Overridepublic void complete(OrderContext context) {throw new UnsupportedOperationException("非法操作");}
}

已支付状态

public class PaidState implements OrderState {@Overridepublic void ship(OrderContext context) {System.out.println("商品已发货");context.setState(new ShippedState());}@Overridepublic void cancel(OrderContext context) {System.out.println("发起退款流程");context.setState(new CancelledState());}// 其他操作处理@Overridepublic void pay(OrderContext context) {throw new UnsupportedOperationException("已支付状态不能重复支付");}// ... 类似处理其他不支持操作
}

其他状态类(已发货、已收货、已完成、已取消)实现方式类似,此处省略。

3. 定义订单上下文

public class OrderContext {private OrderState currentState;public OrderContext() {this.currentState = new UnpaidState(); // 初始状态}public void setState(OrderState state) {this.currentState = state;}// 委托操作给当前状态public void pay() { currentState.pay(this); }public void cancel() { currentState.cancel(this); }public void ship() { currentState.ship(this); }public void receive() { currentState.receive(this); }public void complete() { currentState.complete(this); }
}

4. 客户端使用

public class Client {public static void main(String[] args) {OrderContext order = new OrderContext();order.pay();       // 正常支付order.ship();       // 正常发货order.receive();    // 正常收货order.complete();   // 完成订单try {order.ship();   // 尝试非法操作} catch (Exception e) {System.out.println("操作失败: " + e.getMessage());}}
}

输出结果
复制
支付成功
商品已发货
确认收货成功
订单已完成
操作失败: 已完成状态不能发货

模式优势

优势传统条件分支实现状态模式实现
可维护性修改状态逻辑需改动大量条件判断只需修改对应状态类
可扩展性新增状态需修改所有相关方法新增状态类即可
代码清晰度一个方法包含所有状态逻辑每个状态逻辑独立封装
单一职责违反单一职责原则每个状态类职责明确

实际应用场景

工作流引擎
审批流程中的不同状态(起草、审批中、已通过、被驳回)
游戏开发
角色状态(站立、移动、攻击、死亡)
硬件控制
电梯运行状态(停止、运行、故障)
UI交互
按钮的不同状态(正常、禁用、悬停、点击)

通过状态模式,我们实现了:
✅ 消除复杂的条件判断
✅ 状态转换逻辑内聚到状态类中
✅ 符合开闭原则(新增状态无需修改现有代码)
✅ 更易维护的清晰代码结构

一句话总结

状态模式就是将状态都独立出来,将状态与实际业务解耦。相对于策略模式,状态与状态之间有联系,策略与策略之间是独立的。

相关文章:

  • 人脸检测-人脸关键点-人脸识别-人脸打卡-haar-hog-cnn-ssd-mtcnn-lbph-eigenface-resnet
  • 如何将 ESP32 快速接入高德、心知、和风天气API 获取天气信息
  • void MainWindow::on_btnOutput_clicked()为什么我在QT里面没有connect,也能触发点击效果
  • 【正点原子STM32MP257连载】第四章 ATK-DLMP257B功能测试——RTC时钟测试 #内部RTC时钟 #外部时钟模块AT8563
  • 运维面试题(十四)
  • 常见编码面试问题
  • 命令模式 (Command Pattern)
  • 问题记录(四)——拦截器“失效”?null 还是“null“?
  • 【iOS】OC高级编程 iOS多线程与内存管理阅读笔记——自动引用计数(一)
  • C++ 核心进阶
  • 探秘串口服务器厂家:背后的故事与应用
  • 深入理解Java缓冲输入输出流:性能优化的核心武器
  • 03(总)-docker篇 Dockerfile镜像制作(jdk,jar)与jar包制作成docker容器方式
  • 区块链如何为农业供应链赋能?用 Python 打造透明高效的农产品流通体系
  • Spring Boot 项目的启动流程,图片+文字详细解答(附相关面试题)
  • 进程与线程
  • 如何让Windows开机时自动运行LabVIEW程序
  • 驱动开发硬核特训 · Day 13:从 device_create 到 sysfs,设备文件是如何生成的?
  • OpenCV 图形API(38)图像滤波-----Sobel 算子操作函数Sobel()
  • OpenCv高阶(五)——SIFT特征提取
  • 网站建设与管理单招/肇庆seo外包公司
  • 建立网站 英语怎么说/百度教育官网
  • 做淘宝客网站性质/刷关键词的平台
  • 深圳商城网站制作公司/淘宝店铺怎么运营
  • wordpress 不显示顶部/seo是一种利用搜索引擎
  • 哪家公司网站做的比较好/seo快速排名培训