您好!
欢迎来到京东云开发者社区
登录
首页
博文
课程
大赛
工具
用户中心
开源
首页
博文
课程
大赛
工具
开源
更多
用户中心
开发者社区
>
博文
>
有限状态机在国际计费中的应用探索
分享
打开微信扫码分享
点击前往QQ分享
点击前往微博分享
点击复制链接
有限状态机在国际计费中的应用探索
自猿其说Tech
2022-02-23
IP归属:未知
298680浏览
计算机编程
今天的话题,我们从一个案例开始谈起。 国际计费系统会定期自动生成账单,然后每个账单会按照预设的规则自动进入结算流程,账单从生成之后到结算完成,这期间需要销售支持、结算岗、客户(商家或服务商)、财务、资金等多个不同岗位角色的人员共同参与处理,每个角色处理的环节和操作内容不同,账单的状态也持续发生着改变。 ### 1 为什么要使用状态机 下面这张图,描述了海外应收账单整个生命周期内的全部状态,以及每个状态下可以进行哪些操作行为。 ![](//img1.jcloudcs.com/developer.jdcloud.com/8ac2b797-ad45-4870-ac0d-6ff7c1e3198e20220223154705.png) 对着这张图,我们思考一个问题,在“客户已确认”状态下,能否进行“运营作废”操作呢? 从图中可以看出,“客户已确认”方框上只有一个出发箭头“推送结算”,就是说这个状态下,只能进行“推送结算”这一个操作,因此“客户已确认”状态下是不允许操作“运营作废”的。 这一点,从业务角度很好理解,如果一个账单已经让商家确认完毕,这时候我们再把它作废掉,后续势必涉及让商家重新确认,这对商家来说体验是不好的。 那我们在开发系统时,怎样才能避免这种情况发生呢? 有很多种方式可以实现,比如说,我们采用if判断,代码示例如下: ``` if (状态=“客户已确认”){ if (操作行为=“推送结算”){ pushToSettle(); } else { throw new UnsupportedOperationException(“客户已确认状态下不能操作除推送结算以外的其他操作”); } } else if (状态=其他XXX){ 其他判断处理… } ``` 这种方式实现起来最简单,但是存在的问题也较为明显: 1. 难以通过代码直观体现出“当前状态-操作行为-变更后的新状态这”3者之间的对应关系; 1. 当状态增加或减少时,要修改if-else代码块,当状态和操作行为较多时,容易改错; 1. 如果开发不规范,把这种涉及状态管理的逻辑放到了前端去控制,不仅会使得前端逻辑复杂,还会导致实体状态不一致的严重风险; 我们可以考虑通过状态机来实现,这是一种更加有效稳妥的方式。 **那么什么是状态机呢?** 通常讨论的都是有限状态机。是表示有限个状态以及在这些状态之间的转移和动作等行为的数学模型。 (以下截图来自zhihu.com) ![](//img1.jcloudcs.com/developer.jdcloud.com/f15ba9de-ac4e-441b-ae9b-d7bb20bf274620220223154901.png) 其实,上面描述账单状态变化的这张图就是一个状态机。通过状态机可以集中、统一、规范地管理实体的状态变化。这种管理方式应用非常广泛也很成熟,比如程序代码编译、正则表达式、电子电器设备等领域。 ![](//img1.jcloudcs.com/developer.jdcloud.com/7ed1c6c0-cb19-4ac8-840a-a657e3f217a220220223154918.png) ### 2 主流状态机实现都有哪些,为什么自己开发 最开始需要用状态机时,首先想到的是,这种通用性的东西一定有现成的成熟开源框架。于实网上搜了一番,的确找到很多内容。有教你如何用switch方式写出比if-else更加优雅代码的,有利用枚举值做判断实现的,以及Spirng子项目Spring State Machine。 ![](//img1.jcloudcs.com/developer.jdcloud.com/7b163e64-d169-4375-aaf7-730dafa8400820220223155104.png) 首先说switch或枚举判断的方式,这种方式的问题在于框架性代码与状态配置代码紧密耦合在一起,对于有代码洁癖的我,将不同职能的代码混在一起我是难以接受的。 那按说Spring提供的框架总该可以吧,没错,Spirng State Machine(简称SSM)在抽象层次、规范化、理解方面表现都很出色。但是,由于功能过于强大,导致对于简单的场景来说使用起来有些繁琐,有一种杀鸡用牛刀的感觉。 下面从Spring State Machine项目官网帮助文档中截取了一张图,通过目录中的关键词可以直观感受一下使用SSM的门槛。 ![](//img1.jcloudcs.com/developer.jdcloud.com/d2d63467-b193-4d05-b6d3-d27ab785e4ea20220223155125.png) 本文一开始给出的应收账单状态机,看着似乎有一点点复杂。但是在实际的程序开发中,要实现这个状态机,只需要用到最简单的状态机类型和最基本的概念及特性即可。 因此,决定来开发一个适合自己当前需求的轻量级有限状态机框架(SimpleFSMFrame)。 ### 3 设计思路及关键点 #### 3.1 产品设计目标 一般的状态管理场景,对于状态机的主要诉求只有2点: 1. 判定在某个状态(State)下是否允许进行某个指定的操作行为(Event); 1. 反馈在某个状态(State)下都允许进行哪些操作行为(Event); 对于更加复杂的场景,不在本次设计考虑范围内,将作为未来扩展的方向。 #### 3.2 技术实现目标 既然定位成框架,那么就需要具备以下特性: 1. 可复用,该框架可以开源或者以jar包形式提供给别人使用; 1. 简单易用,只需了解状态机最基本的3个概念即可:State(状态)、Event(事件)、Transition(转换); 1. 与业务无关,框架本身只实现状态机本身的基本概念和功能特性,不包含任何具体实体的状态转换关系管理,也就是说不能对使用者产生干扰。 1. 能扩展,模块粒度以及层级拆分合理,高内聚低耦合 #### 3.3 框架详细设计 ![](//img1.jcloudcs.com/developer.jdcloud.com/9776426c-31fd-4242-9785-b47152f28e8b20220223155226.png) - **组件1:StateMachine 状态机接口** 定义了状态机的行为,包含了上述2个诉求点。 ```java /** * 在当前状态下执行某个事件 * * @param event 事件 * @return 若执行成功则返回变更后的新状态 * @throws UnsupportedOperationException 如果当前状态不支持该操作则抛出此异常 */ State onEvent(Event event) throws UnsupportedOperationException; /** * 当前的状态 * * @return */ State getState(); /** * 当前状态可执行的事件清单 * * @return */ List<Event> acceptableEvents(); /** * 当前状态是否可以执行指定的某个事件(仅给出是否允许执行的判断结论,不会真的执行) * * @param event 事件 * @return */ boolean canPerformEvent(Event event); ``` - **组件2:State 状态接口** 规范了作为“状态”概念的对象应当具备的最基本的行为。 - **组件3:Event 事件接口** 规范了作为“事件”概念的对象应当具备的最基本的行为 - **组件4:Transition 状态转换关系接口** 定义了在一个条状态与事件的转换关系中,哪些对象应当参与其中以及各个对象在其中所扮演的角色。 - **组件5:SimpleFSMFrame 轻量级有限状态机框架** 提供状态机基本概念与行为的实现。使用者只需继承此类即可实现一个状态机实例。 ##### 关键设计 首先看这个类的构造方法: ```java /** * 初始化一个状态机 * * @param initialState 初始状态 * @param transitions 状态与事件之间的转换关系 */ public SimpleFSMFrame(State initialState, Transition[] transitions) { state = initialState; this.transitionBox = new TransitionBox(transitions); } ``` 构造方法要求必须传入一个初始状态,这个参数在创建状态机时直接可以把状态置为指定的初始状态,而不必让状态机从真正的初始状态开始,避免了类似SSM中需要先对状态机本身进行序列化以及持久化,然后再反序列化恢复状态的繁杂过程。 对于状态机中最为关键,对于框架程序来说最需要解耦的部分,即状态转换关系配置部分,是整个设计中的重中之重。需要考虑灵活易配置、来源方式开放、对框架程序无任何耦合这几个目标。 因此在构造方法的第二个参数中,要求传入该状态机的完整转换关系,形式为数组。用户程序(即继承此类的子类)可以按照自己最方便的方式来“整理”状态转换关系。比如,将状态转换关系存到数据库中,构建状态机时从数据库中读出来即可;再比如,通过专门的图形化状态机绘制工具将画好的状态机图形转换为这里要求的数组数据,以便构造一个新的状态机。因此对于状态关系的配置方式是支持扩展的。 但是这里之所以设计为数组形式,其实是有另有考虑的。可以用枚举enum来定义状态转换关系,然后用values()方法就能轻松获取到全部的转换关系了,而且是数组形式。——利用了java语言的特性,如果是非java语言可以考虑类似方式。 下面给出这个类的详细代码: ```java import java.util.*; import java.util.stream.Collectors; /** * 轻量级的状态机框架,通过集成此类可快速实现一个简易的有限状态机。 * <br> * 线程安全 * * @author xieyipei * @date 2021/8/13 18:13 */ public class SimpleFSMFrame implements StateMachine { /** * 存放有当前状态机中的状态与事件转换关系的box */ private final TransitionBox transitionBox; /** * 状态机当前状态 */ private State state; /** * 初始化一个状态机 * * @param initialState 初始状态 * @param transitions 状态与事件之间的转换关系 */ public SimpleFSMFrame(State initialState, Transition[] transitions) { state = initialState; this.transitionBox = new TransitionBox(transitions); } @Override synchronized public State onEvent(Event event) throws UnsupportedOperationException { state = execute(state, event); return state; } @Override public State getState() { return state; } @Override public List<Event> acceptableEvents() { return acceptableEvents(state); } @Override public boolean canPerformEvent(Event event) { return canPerformEvent(state, event); } /** * 在指定状态下执行某个事件,执行成功返回变更后的新状态 * * @param currentState 状态 * @param event 事件 * @return 变更后的新状态 * @throws UnsupportedOperationException 如果当前状态不支持该操作则抛出此异常 */ private State execute(State currentState, Event event) throws UnsupportedOperationException { List<Transition> transitions = transitionBox.getTransitionBySource(currentState); return transitions .stream() .filter(transition -> transition.getEvent().equals(event)) .findAny() .orElseThrow(() -> new UnsupportedOperationException("Event:" + event.name() + " can not be performed on State:" + currentState.name())) .getTarget(); } /** * 当前状态可执行的事件清单 * * @param state 状态 * @return */ private List<Event> acceptableEvents(State state) { List<Transition> transitions = transitionBox.getTransitionBySource(state); return transitions .stream() .map(transition -> transition.getEvent()) .collect(Collectors.toList()); } /** * 当前状态是否可以执行指定的某个事件(仅给出是否允许执行的判断结论,不会真的执行) * * @param state 状态 * @param event 事件 * @return */ private boolean canPerformEvent(State state, Event event) { List<Transition> transitions = transitionBox.getTransitionBySource(state); return transitions .stream() .anyMatch(transition -> transition.getEvent().equals(event)); } /** * 检验状态与事件转换关系是否合法 * * @param transitions * @throws IllegalArgumentException 如果校验不通过则抛出此异常 */ private void verifyTransition(Transition[] transitions) throws IllegalArgumentException { //检查源状态+事件不能重复 Set<String> set = new HashSet<>(); for (Transition transition : transitions) { String key = transition.getSource().name() + "" + transition.getEvent().name(); boolean flag = set.add(key); if (!flag) throw new IllegalArgumentException(String.format("reduplicate transition source=%s event=%s", transition.getSource().name(), transition.getEvent().name())); } } /** * 存放整理后的状态与事件转换关系,并提供相应的访问方法 */ private class TransitionBox { private Map<State, List<Transition>> sourceMap = new HashMap<>(); private Map<State, List<Transition>> targetMap = new HashMap<>(); private Map<Event, List<Transition>> eventMap = new HashMap<>(); /** * 根据状态与事件的转换关系初始化一个box * * @param transitions 状态与事件的转换关系 */ public TransitionBox(Transition[] transitions) { //校验转换关系是否存在异常情况,如果存在则抛出异常 verifyTransition(transitions); for (Transition transition : transitions) { //sourceMap List<Transition> sourceList = sourceMap.get(transition.getSource()); if (sourceList == null) { sourceList = new ArrayList<>(); sourceMap.put(transition.getSource(), sourceList); } sourceList.add(transition); //targetMap List<Transition> targetList = targetMap.get(transition.getTarget()); if (targetList == null) { targetList = new ArrayList<>(); targetMap.put(transition.getTarget(), targetList); } targetList.add(transition); //eventMap List<Transition> eventList = eventMap.get(transition.getEvent()); if (eventList == null) { eventList = new ArrayList<>(); eventMap.put(transition.getEvent(), eventList); } eventList.add(transition); } } /** * 获取指定源状态的所有转换关系 * * @param source 源状态 * @return */ public List<Transition> getTransitionBySource(State source) { List<Transition> list = sourceMap.get(source); return list != null ? list : new ArrayList<>(); } /** * 获取指定目标状态的所有转换关系 * * @param target 目标状态 * @return */ public List<Transition> getTransitionByTarget(State target) { List<Transition> list = targetMap.get(target); return list != null ? list : new ArrayList<>(); } /** * 获取与指定事件相关的所有转换关系 * * @param event 事件 * @return */ public List<Transition> getTransitionByEvent(Event event) { List<Transition> list = eventMap.get(event); return list != null ? list : new ArrayList<>(); } } } ``` 整体思路是,将构造方法传入的所有状态转换关系放到定义为私有内部类TransitionBox这样一个容器中保管,避免对外暴露内部实现细节,在TransitionBox中会对关系配置进行校验,以及整理为3个不同的map,并通过这些map实现状态机的行为判断。 ![](//img1.jcloudcs.com/developer.jdcloud.com/efae08bc-4817-4b00-bf74-eab0a04c059f20220223155500.png) ### 4 使用案例 #### 4.1 定义状态机 对于使用者来说,只需3步即可完成一个全新的状态机实现: 1. 实现State和Event接口,定义自己的状态和事件; 1. 定义枚举类并实现Transition接口,状态转换关系通过枚举值形式配置出来; 1. 继承SimpleFSMFrame类,调用上一步枚举类的values()方法并传入构造方法; 下面给出一个项目中实际使用的案例: ```java /** * 适用于海外应收账单状态(相比跨境应收增加了3个新状态) * * @author xieyipei * @date 2021/9/23 14:57 */ public class ARBillStateMachine extends SimpleFSMFrame { /** * 初始化一个状态机 * * @param initialState 初始状态 */ public ARBillStateMachine(State initialState) { //调用自定义的状态转换关系枚举的values()方法获取到全部转换关系,然后传给父类的构造方法 super(initialState, ARTransition.values()); } @Getter private enum ARTransition implements Transition { //状态转换关系通过枚举值形式配置出来。形式为:sourceState+event+targetState T111(BillState.INIT, BillEvent.CONFIRM, BillState.MERCHANT_CLIENT_PENDING), T121(BillState.INIT, BillEvent.DISCARD, BillState.DISCARDED), T211(BillState.MERCHANT_CLIENT_PENDING, BillEvent.CLIENT_REJECT, BillState.OPERATING_PENDING), T212(BillState.MERCHANT_CLIENT_PENDING, BillEvent.MERCHANT_CLIENT_REJECT, BillState.OPERATING_PENDING), T213(BillState.MERCHANT_CLIENT_PENDING, BillEvent.ON_BEHALF_OF_CLIENT_REJECT, BillState.OPERATING_PENDING), T214(BillState.MERCHANT_CLIENT_PENDING, BillEvent.ON_BEHALF_OF_MERCHANT_CLIENT_REJECT, BillState.OPERATING_PENDING), T221(BillState.MERCHANT_CLIENT_PENDING, BillEvent.CLIENT_ACCEPT, BillState.MERCHANT_CLIENT_CONFIRMED), T222(BillState.MERCHANT_CLIENT_PENDING, BillEvent.MERCHANT_CLIENT_ACCEPT, BillState.MERCHANT_CLIENT_CONFIRMED), T223(BillState.MERCHANT_CLIENT_PENDING, BillEvent.ON_BEHALF_OF_CLIENT_ACCEPT, BillState.MERCHANT_CLIENT_CONFIRMED), T224(BillState.MERCHANT_CLIENT_PENDING, BillEvent.ON_BEHALF_OF_MERCHANT_CLIENT_ACCEPT, BillState.MERCHANT_CLIENT_CONFIRMED), T311(BillState.OPERATING_PENDING, BillEvent.DISCARD, BillState.DISCARDED), T321(BillState.OPERATING_PENDING, BillEvent.CONFIRM, BillState.MERCHANT_CLIENT_PENDING), T411(BillState.MERCHANT_CLIENT_CONFIRMED, BillEvent.PUSH_TO_SETTLE, BillState.SETTLEMENT_PENDING), T421(BillState.MERCHANT_CLIENT_CONFIRMED, BillEvent.DISCARD, BillState.DISCARDED), T511(BillState.SETTLEMENT_PENDING, BillEvent.PARTIAL_PAYMENT_WAS_RECEIVED, BillState.PARTIAL_PAYMENT_WAS_RECEIVED), T521(BillState.SETTLEMENT_PENDING, BillEvent.OPERATING_CANCEL, BillState.FINANCIAL_REJECTED), T522(BillState.SETTLEMENT_PENDING, BillEvent.FINANCIAL_REJECT, BillState.FINANCIAL_REJECTED), T523(BillState.SETTLEMENT_PENDING, BillEvent.REJECT_IN_SETTLEMENT, BillState.FINANCIAL_REJECTED), T531(BillState.SETTLEMENT_PENDING, BillEvent.COMPLETE_SETTLEMENT, BillState.SETTLEMENT_FINISHED), T533(BillState.SETTLEMENT_PENDING, BillEvent.PUSH_TO_SETTLE, BillState.SETTLEMENT_PENDING), T611(BillState.PARTIAL_PAYMENT_WAS_RECEIVED, BillEvent.FULL_PAYMENT_WAS_RECEIVED, BillState.SETTLEMENT_FINISHED), T612(BillState.PARTIAL_PAYMENT_WAS_RECEIVED, BillEvent.PARTIAL_PAYMENT_WAS_RECEIVED, BillState.PARTIAL_PAYMENT_WAS_RECEIVED), T613(BillState.PARTIAL_PAYMENT_WAS_RECEIVED, BillEvent.COMPLETE_SETTLEMENT, BillState.SETTLEMENT_FINISHED), T711(BillState.FINANCIAL_REJECTED, BillEvent.DISCARD, BillState.DISCARDED), ; private final State source; private final State target; private final Event event; ARTransition(State source, Event event, State target) { this.source = source; this.target = target; this.event = event; } } } ``` #### 4.2 使用状态机 ```java private boolean canPerformEvent(Bill bill, BillEvent billEvent) { //根据账单状态初始化状态机 StateMachine stateMachine = new ARBillStateMachine(bill.getBillState()); //通过状态机判断是否允许操作指定的行为 return stateMachine.canPerformEvent(billEvent); } ``` ### 5 改进空间讨论 **分层多级状态如何支持?** 例如,账单第一级状态可分为,初始、客户确认中、待结算、完成。其中待结算状态又细分二级状态为:已推送结算、财务审批通过、资金撤单、结算完成。这样,状态之间不再是简单的互不包含,而是存在包含关系,也就是出现了复合状态。 针对这个问题,大家是如何看的,欢迎讨论~ ------------ ###### 自猿其说Tech-京东物流技术发展部 ###### 作者:谢益培
原创文章,需联系作者,授权转载
上一篇:一个低代码报表配置平台实践
下一篇:Shell在日常工作中的应用实践
相关文章
Taro小程序跨端开发入门实战
Flutter For Web实践
配运基础数据缓存瘦身实践
自猿其说Tech
文章数
426
阅读量
2149963
作者其他文章
01
深入JDK中的Optional
本文将从Optional所解决的问题开始,逐层解剖,由浅入深,文中会出现Optioanl方法之间的对比,实践,误用情况分析,优缺点等。与大家一起,对这项Java8中的新特性,进行理解和深入。
01
Taro小程序跨端开发入门实战
为了让小程序开发更简单,更高效,我们采用 Taro 作为首选框架,我们将使用 Taro 的实践经验整理了出来,主要内容围绕着什么是 Taro,为什么用 Taro,以及 Taro 如何使用(正确使用的姿势),还有 Taro 背后的一些设计思想来进行展开,让大家能够对 Taro 有个完整的认识。
01
Flutter For Web实践
Flutter For Web 已经发布一年多时间,它的发布意味着我们可以真正地使用一套代码、一套资源部署整个大前端系统(包括:iOS、Android、Web)。渠道研发组经过一段时间的探索,使用Flutter For Web技术开发了移动端可视化编程平台—Flutter乐高,在这里希望和大家分享下使用Flutter For Web实践过程和踩坑实践
01
配运基础数据缓存瘦身实践
在基础数据的常规能力当中,数据的存取是最基础也是最重要的能力,为了整体提高数据的读取能力,缓存技术在基础数据的场景中得到了广泛的使用,下面会重点展示一下配运组近期针对数据缓存做的瘦身实践。
自猿其说Tech
文章数
426
阅读量
2149963
作者其他文章
01
深入JDK中的Optional
01
Taro小程序跨端开发入门实战
01
Flutter For Web实践
01
配运基础数据缓存瘦身实践
添加企业微信
获取1V1专业服务
扫码关注
京东云开发者公众号