【设计模式】订单状态流传中的状态机与状态模式

文章目录

  • 1. 前言
  • 2.状态模式
    • 2.1.订单状态流转案例
      • 2.1.1.状态枚举定义
      • 2.1.2.状态接口与实现
      • 2.1.3.状态机
      • 2.1.4.测试
    • 2.2.退款状态的拓展
      • 2.2.1.代码拓展
      • 2.2.2.测试
    • 2.3.小结
  • 3.总结

1. 前言

状态模式一般是用在对象内部的状态流转场景中,用来实现状态机

什么是状态机呢?
状态机是对状态转移的抽象,由事件状态动作组成,事件有时候也被称为转移事件或者转移,当事件触发时,可以将状态由一个状态变更为另一个状态,并执行动作。其中,事件和状态是必须存在的,动作可以不要。

下面是一张状态图,表达的就是一个状态机的模型。
在这里插入图片描述
通俗来讲,就是对状态的变更做了一定的限制,不能随意的修改状态,而是只有处于某个特定的状态时,才能变更到另一个特定的状态。

2.状态模式

状态模式将状态抽象成一个个的状态对象,状态机当前持有某个状态对象,就表示当前的状态机处于什么状态。
然后将事件处理为一个个的方法,每个方法中会操作状态机修改状态,有需要的情况下,在修改状态的同时还可以执行某些动作。

把通用部分提取出来后,可以得到这样一个通用类图:
在这里插入图片描述

可以看到上面的StateMachineState关系是双向的,这是因为状态机需要持有状态对象来表示当前状态,以及通过当前的状态对象中的方法进行状态的流转,而流转的结果需要重新set到状态机中,又要求State必须持有状态机对象。
当然,这里StateStateMachine的关系也可以通过依赖来表示。

2.1.订单状态流转案例

假设现在有一个商品订单的状态流转需求,状态图如下:
在这里插入图片描述
这里没有加退款的状态,后续的拓展例子上会加上,用这种方式来体验状态模式的拓展性。

我们拿着这个图的时候,可以简单的在脑海里面过一遍如果通过if/else或者switch来做,应该要怎么写,后续如果想把退款的状态加入进去又该怎么拓展,这种方式应该大家都会,就不在这里赘述了。

接下来,就一步步的通过状态模式来实现这么一个状态机。

2.1.1.状态枚举定义

定义状态枚举主要是为了统一状态常量,因为订单是需要落库的,我们在持久化到数据库时,不能把状态对象保存进去,所以会涉及到状态常量与状态对象的互相转换。定义的枚举如下:

import lombok.Getter;@Getter
public enum OrderStateEnum {WAIT_PAYMENT(1, "待支付"),WAIT_DELIVER(2, "待发货"),WAIT_RECEIVE(3, "待收货"),RECEIVED(4, "已收货"),CANCEL(5, "已取消");private final int state;private final String desc;OrderStateEnum(int state, String desc) {this.state = state;this.desc = desc;}public int getState() {return state;}public String getDesc() {return desc;}
}

2.1.2.状态接口与实现

先上代码:

public interface OrderState {OrderStateEnum orderStateType();default void pay(OrderStateMachine stateMachine) {System.out.println("|--当前订单状态不支持支付,已忽略");}default void cancel(OrderStateMachine stateMachine) {System.out.println("|--当前订单状态不支持取消,已忽略");}default void deliver(OrderStateMachine stateMachine) {System.out.println("|--当前订单状态不支持发货,已忽略");}default void receive(OrderStateMachine stateMachine) {System.out.println("|--当前订单状态不支持收货,已忽略");}
}

接口中定义的pay,cancel等方法就是事件,供子类进行实现,相信大家也发现了,这些事件没有定义成抽象方法,而是通过default定义成了一个实例方法。不太清楚为什么的同学,可以先思考一下为什么要这么定义。

其实这么定义的好处是各个状态子类只需要实现自己需要的方法,而不用把所有的方法都实现一遍,这种做法在Spring中也比较常见,在JDK8之前通常是用xxxWrapper来实现的,JDK8之后就重构为直接使用default方法来实现了。

举个例子:后续如果需要加入退款状态,接口中也会新增一个提交退款的事件,在各个子类中,选择需要实现提交退款事件的状态子类进行重写即可,而不需要所有的子类都重写。


有多少个状态,就有多少个实现类,并按照上面的状态图,在对应的状态中实现自己需要的事件。

  • 待支付状态:有支付和取消两种事件
    public class WaitPaymentState implements OrderState {@Overridepublic OrderStateEnum orderStateType() {return OrderStateEnum.WAIT_PAYMENT;}@Overridepublic void pay(OrderStateMachine stateMachine) {stateMachine.setCurrentState(new WaitDeliverState());}@Overridepublic void cancel(OrderStateMachine stateMachine) {stateMachine.setCurrentState(new CancelState());}
    }
    
  • 待发货状态:有发货事件
    public class WaitDeliverState implements OrderState {@Overridepublic OrderStateEnum orderStateType() {return OrderStateEnum.WAIT_DELIVER;}@Overridepublic void deliver(OrderStateMachine stateMachine) {stateMachine.setCurrentState(new WaitReceiveState());}
    }
    
  • 待收货状态:有收货事件
    public class WaitReceiveState implements OrderState {@Overridepublic OrderStateEnum orderStateType() {return OrderStateEnum.WAIT_RECEIVE;}@Overridepublic void receive(OrderStateMachine stateMachine) {stateMachine.setCurrentState(new ReceivedState());}
    }
    
  • 已收货状态:状态结束点,没有其他事件
    public class ReceivedState implements OrderState {@Overridepublic OrderStateEnum orderStateType() {return OrderStateEnum.RECEIVED;}}
    
  • 取消状态:状态结束点,没有其他事件
    public class CancelState implements OrderState {@Overridepublic OrderStateEnum orderStateType() {return OrderStateEnum.CANCEL;}
    }
    

2.1.3.状态机

状态机中需要持有当前状态对象,同时需要把状态接口中的事件同步定义到状态机中,以便外部业务对象调用。
除此之外,状态枚举常量与状态对象之间的映射关系也可以直接配置在当前状态机中,功能更加内聚。

public class OrderStateMachine {public static final Map<OrderStateEnum, OrderState> ORDER_STATE_MAP = new HashMap<>();static {ORDER_STATE_MAP.put(OrderStateEnum.WAIT_PAYMENT, new WaitPaymentState());ORDER_STATE_MAP.put(OrderStateEnum.WAIT_DELIVER, new WaitDeliverState());ORDER_STATE_MAP.put(OrderStateEnum.WAIT_RECEIVE, new WaitReceiveState());ORDER_STATE_MAP.put(OrderStateEnum.RECEIVED, new ReceivedState());ORDER_STATE_MAP.put(OrderStateEnum.CANCEL, new CancelState());}private OrderState currentState;public OrderStateMachine(OrderStateEnum orderStateEnum) {this.currentState = ORDER_STATE_MAP.get(orderStateEnum);}public OrderState getCurrentState() {return currentState;}public void setCurrentState(OrderState currentState) {this.currentState = currentState;}void pay() {currentState.pay(this);}void deliver() {currentState.deliver(this);}void receive() {currentState.receive(this);}void cancel() {currentState.cancel(this);}}

2.1.4.测试

做一下状态机的测试,由于打印的日志重复度很高,这里取了个巧,将函数作为参数封装了一下:

public class OrderService {public static void main(String[] args) {OrderStateMachine stateMachine = new OrderStateMachine(OrderStateEnum.WAIT_DELIVER);invoke(stateMachine::pay, "用户支付", stateMachine);invoke(stateMachine::deliver, "商家发货", stateMachine);invoke(stateMachine::receive, "用户收货", stateMachine);invoke(stateMachine::cancel, "取消支付", stateMachine);}public static void invoke(Runnable runnable, String desc, OrderStateMachine stateMachine) {System.out.println(desc + "前订单状态: " + stateMachine.getCurrentState().orderStateType().getDesc());runnable.run();System.out.println(desc + "后订单状态: " + stateMachine.getCurrentState().orderStateType().getDesc());System.out.println("------------------");}
}

待发货作为状态常量创建了一个状态机,状态机当前的状态就是待发货,下面的四个调用中,第1,4个是不会改变状态的,第2,3个会改变状态,下面以执行结果来验证猜测:

用户支付前订单状态: 待发货
|--当前订单状态不支持支付,已忽略
用户支付后订单状态: 待发货
------------------
商家发货前订单状态: 待发货
商家发货后订单状态: 待收货
------------------
用户收货前订单状态: 待收货
用户收货后订单状态: 已收货
------------------
取消支付前订单状态: 已收货
|--当前订单状态不支持取消,已忽略
取消支付后订单状态: 已收货
------------------

2.2.退款状态的拓展

通过状态模式来实现状态机,看重的就是它带来的拓展性和易维护性,所以在原有的基础上,加上退款的事件和状态,一起看看需要做些什么事。

2.2.1.代码拓展

下面是加入了退款的状态图:
在这里插入图片描述
通过状态图可以看到,需要加入:

  • 两个状态:退款中已退款
  • 两个事件:申请退款确认退款
  • 原有状态拓展:待发货、待收货、已收货 3个状态中都需要引入申请退款事件

综上,一步一步的拓展代码:

  • 第一步:拓展枚举常量
    public enum OrderStateEnum {WAIT_PAYMENT(1, "待支付"),WAIT_DELIVER(2, "待发货"),WAIT_RECEIVE(3, "待收货"),RECEIVED(4, "已收货"),CANCEL(5, "已取消"),REFUNDING(6, "退款中"),REFUNDED(7, "已退款"),;// 省略后续代码……
    }
    
  • 第二步:拓展状态接口
    public interface OrderState {// 省略已有代码……default void refund(OrderStateMachine stateMachine) {System.out.println("|--当前订单状态不支持退款,已忽略");}default void confirmRefund(OrderStateMachine stateMachine) {System.out.println("当前订单状态不支持确认退款,已忽略");}
    }
    
  • 第三步:新增两个状态,退款中已退款
public class RefundingState implements OrderState {@Overridepublic OrderStateEnum name() {return OrderStateEnum.REFUNDING;}@Overridepublic void confirmRefund(OrderStateMachine stateMachine) {stateMachine.setCurrentState(new RefundedState());}
}
public class RefundedState implements OrderState {@Overridepublic OrderStateEnum name() {return OrderStateEnum.REFUNDED;}
}
  • 第四步:拓展原有状态,待发货待收货已收货
public class WaitDeliverState implements OrderState {// 省略已有代码……@Overridepublic void refund(OrderStateMachine stateMachine) {stateMachine.setCurrentState(new RefundingState());}
}
public class WaitReceiveState implements OrderState {// 省略已有代码……@Overridepublic void refund(OrderStateMachine stateMachine) {stateMachine.setCurrentState(new RefundingState());}}
public class ReceivedState implements OrderState {// 省略已有代码……@Overridepublic void refund(OrderStateMachine stateMachine) {stateMachine.setCurrentState(new RefundingState());}}
  • 第五步:拓展状态机
public class OrderStateMachine {public static final Map<OrderStateEnum, OrderState> ORDER_STATE_MAP = new HashMap<>();static {// 省略已有状态……ORDER_STATE_MAP.put(OrderStateEnum.REFUNDING, new RefundingState());ORDER_STATE_MAP.put(OrderStateEnum.REFUNDED, new RefundedState());}// 省略已有方法……void refund() {currentState.refund(this);}void confirmRefund() {currentState.confirmRefund(this);}
}

2.2.2.测试

在上面的代码中可以看到,都是在对配置进行追加,而没有对原有的逻辑做任何的修改,然后写一个测试:

public class OrderService {public static void main(String[] args) {OrderStateMachine stateMachine = new OrderStateMachine(OrderStateEnum.WAIT_DELIVER);invoke(stateMachine::pay, "用户支付", stateMachine);invoke(stateMachine::deliver, "商家发货", stateMachine);invoke(stateMachine::receive, "用户收货", stateMachine);invoke(stateMachine::cancel, "取消支付", stateMachine);invoke(stateMachine::refund, "申请退款", stateMachine);invoke(stateMachine::confirmRefund, "确认退款", stateMachine);}public static void invoke(Runnable runnable, String desc, OrderStateMachine stateMachine) {System.out.println(desc + "前订单状态: " + stateMachine.getCurrentState().orderStateType().getDesc());runnable.run();System.out.println(desc + "后订单状态: " + stateMachine.getCurrentState().orderStateType().getDesc());System.out.println("------------------");}}

查看日志,是否触发退款:

用户支付前订单状态: 待发货
|--当前订单状态不支持支付,已忽略
用户支付后订单状态: 待发货
------------------
商家发货前订单状态: 待发货
商家发货后订单状态: 待收货
------------------
用户收货前订单状态: 待收货
用户收货后订单状态: 已收货
------------------
取消支付前订单状态: 已收货
|--当前订单状态不支持取消,已忽略
取消支付后订单状态: 已收货
------------------
申请退款前订单状态: 已收货
申请退款后订单状态: 退款中
------------------
确认退款前订单状态: 退款中
确认退款后订单状态: 已退款
------------------

2.3.小结

从上面的代码可以看到,通过状态模式可以很轻松的对状态进行拓展。

不过上面的例子中没有对状态机中的动作进行实现,其实动作和状态转换的逻辑放在一起就可以了,即通过事件(方法调用) 可以变更状态,同时也能够触发对应的动作。

此外,代码中只是状态机的流程,实际的开发中应该将状态机关联到对应的业务实体中,通过业务实体的实时状态来创建状态机,在完成状态流转之后再将状态更新到业务实体中。

3.总结

本篇主要讲述了如何通过状态模式来实现一个状态机。状态模式的实现,代码结构清晰(相对于if/else,switch)拓展性强,同时也起到了良好的封装效果(状态在状态机内部流转,业务流程不需要关心状态到底是怎么流转的)。

当然缺点就是类膨胀问题,类会比较多,如果状态非常复杂的情况下,也可以采取其他办法来实现状态机,例如查表法。

总之,要分析并实现一个业务流程中的状态流转的时候,先画出状态图,以状态图为指导来选择状态机的实现方式即可,在状态相对不那么复杂的情况下,可以优先考虑使用状态模式。

附:《【UML建模】状态图(State Machine Diagram)》

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

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

相关文章

使用percona-xtrabackup备份MySQL数据

xtrabackup备份分为两种 本文参考链接1 本文参考链接2 全量备份 1.备份数据 要创建备份&#xff0c;请xtrabackup使用xtrabackup --backup option. 您还需要指定一个xtrabackup --target-dir选项&#xff0c;即备份的存储位置&#xff0c;如果InnoDB数据或日志文件未存储在同…

今天,谷歌Chrome浏览器部署抗量子密码

谷歌已开始部署混合密钥封装机制&#xff08;KEM&#xff09;&#xff0c;以保护在建立安全的 TLS 网络连接时共享对称加密机密。 8月10日&#xff0c;Chrome 浏览器安全技术项目经理Devon O’Brien解释说&#xff0c;从 8 月 15 日发布的 Chrome 浏览器 116 开始&#xff0c;谷…

SpringBoot集成Solr(一)保存数据到Solr

SpringBoot集成Solr&#xff08;一&#xff09;保存数据到Solr 添加依赖 <!--SpringBoot中封装过的Solr依赖--> <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-solr</artifactId><ver…

数组详解

1. 一维数组的创建和初始化 1.1 数组的创建 数组是一组相同类型元素的集合。 数组的创建方式&#xff1a; type_t arr_name [const_n]; //type_t 是指数组的元素类型 //const_n 是一个常量表达式&#xff0c;用来指定数组的大小 数组创建的实例&#xff1a; //代码1 int a…

STM32 PB9的外部中断

STM32F103中&#xff0c;外部中断的IRQHandler&#xff0c;EXTI0至EXTI1都是单独的&#xff0c;如图1所示&#xff1b;而从EXTI5至9则共用一个&#xff0c;如图2所示。这些可以通过CtrlF在全工程内查找。 图1 图2 以PB9为例&#xff0c;宏定义可如下配置&#xff1a; // 定义…

【Lua】(一)VSCode 搭建 Lua 开发环境

前言 最近在找工作&#xff0c;基本所有的岗位都会问到 Lua&#xff08;甚至拼 UI 的都要求会 Lua&#xff09;&#xff0c;咱能怎么办呢&#xff0c;咱也只能学啊…… 工欲善其事&#xff0c;必先利其器。第一步&#xff0c;先来把环境配置好吧&#xff01; 当前适用版本&a…

图数据库_Neo4j学习cypher语言_常用函数_关系函数_字符串函数_聚合函数_数据库备份_数据库恢复---Neo4j图数据库工作笔记0008

然后再来看一些常用函数,和字符串函数,这里举个例子,然后其他的 类似 可以看到substring字符串截取函数 可以看到截取成功 聚合函数 这里用了一个count(n) 统计函数,可以看到效果 关系函数,我们用过就是id(r) 可以取出对应的r的id来这样..

学习电工有哪些好处?在哪学习电工?

学习电工有哪些好处&#xff1f;在哪学习电工&#xff1f;学习电工可以做什么&#xff1f;优势有哪些&#xff1f; 学习电工可以做什么&#xff1f;学习电工有哪些好处&#xff1f; 就业去向&#xff1a;可在企业单位从事电气设备的安装、调试、操作、维护等工作&#xff0c;…

“保姆级”考研下半年备考时间表

7月-8月 确定考研目标与备考计划 暑假期间是考研复习的关键时期&#xff0c;需要复习的主要内容有&#xff1a;重点关注重要的学科和专业课程&#xff0c;复习相关基础知识和核心概念。制定详细的复习计划并合理安排每天的学习时间&#xff0c;增加真题练习熟悉考试题型和答题技…

【框架类】—MVVM框架

一、MVVM框架有哪些 Vue.jsReact.jsAngular.js 二、对MVVM的认识 1. MVC是什么 全称 Model View Controller, 它采用模型(Model)-视图(View)-控制器(controller)的方法把业务逻辑、数据与界面显示分离 2. MVVM的定义 MVVM是一种软件架构模式&#xff0c;它代表了模型 --视…

聊聊51单片机

目录 1.介绍 2.发展 3.应用领域 4.发展前景 1.介绍 51单片机&#xff08;AT89C51&#xff09;是一种常见的8位微控制器&#xff0c;属于Intel MCS-51系列。它是一种低功耗、高性能的单片机&#xff0c;广泛应用于嵌入式系统中。 51单片机具有很多特点和功能&#xff0c;例如…

26、springboot的自动配置03--核心功能--自定义条件注解及使用

开发自己的自动配置------开发自己的条件注解 ★ 自定义条件注解 好处有两个&#xff1a; 1. 真正掌握Spring boot条件注解的本质。 2. 项目遇到一些特殊的需求时&#xff0c;也可以开发自己的自定义条件注解来解决问题。自定义条件注解&#xff1a; ▲ 所有自定义注解其实都…

uni-app的Vue.js实现微信小程序的紧急事件登记页面功能

主要功能实现 完成发生时间选择功能&#xff0c;用户可以通过日期选择器选择事件发生的时间。实现事件类型选择功能&#xff0c;用户可以通过下拉选择框选择事件的类型。添加子养殖场编号输入框&#xff0c;用户可以输入与事件相关的子养殖场编号。完成事件描述输入功能&#…

【MySQL系列】--初识数据库

&#x1f490; &#x1f338; &#x1f337; &#x1f340; &#x1f339; &#x1f33b; &#x1f33a; &#x1f341; &#x1f343; &#x1f342; &#x1f33f; &#x1f344;&#x1f35d; &#x1f35b; &#x1f364; &#x1f4c3;个人主页 &#xff1a;阿然成长日记 …

[管理与领导-29]:IT基层管理者 - 团队管理 - 马斯洛需求层次在家庭、情侣、朋友、人生、职场中应用

目录 前言&#xff1a; 一、马斯洛需求层次理论 二、马斯洛需求层次理论在家庭中的应用 三、斯洛需求层次在夫妻、情侣之间的应用 四、马斯洛需求层次理论在朋友间的应用 五、马斯洛需求层次理论在人生发展的应用 六、马斯洛需求层次理论在职场中的应用 前言&#xff1a…

消息中间件相关面试题

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱发博客的嗯哼&#xff0c;爱好Java的小菜鸟 &#x1f525;如果感觉博主的文章还不错的话&#xff0c;请&#x1f44d;三连支持&#x1f44d;一下博主哦 &#x1f4dd;社区论坛&#xff1a;希望大家能加入社区共同进步…

Eclipse集成MapStruct

Eclipse集成MapStruct 在Eclipse中添加MapStruct依赖配置Eclipse支持MapStruct①安装 m2e-aptEclipse Marketplace的方式安装Install new software的方式安装&#xff08;JDK8用到&#xff09; ②添加到pom.xml 今天拿到同事其他项目的源码&#xff0c;导入并运行的时候抛出了异…

【二叉树前沿篇】树

【二叉树前沿篇】树 1 树的概念2. 树的相关概念3. 树的表示4. 树在实际中的运用&#xff08;表示文件系统的目录树结构&#xff09; 1 树的概念 树是一种非线性的数据结构&#xff0c;它是由n&#xff08;n>0&#xff09;个有限结点组成一个具有层次关系的集合。把它叫做树是…