【Design Pattern】长流程 - 职责链模式的运用

2024-08-26 20:58

本文主要是介绍【Design Pattern】长流程 - 职责链模式的运用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

【前提 & 场景】

       写业务逻辑的时候(和银行交互或者金融相关业务),小编常会遇到这种情况,如图:

       

        流程比较复杂,订单处理节点会很多,按照传统的设计,代码的条理性、可读性都会很差,经常会出现很多if else判断,不去好好设计,很容易被后人视为“坏味道”  。而“职责链模式”恰恰能很好解决这个问题,详情往下看。

 

【简介】

     1.概念

      职责链 - 使多个对象(处理者)都有机会处理请求,从而减缓了请求的发送者和(每个)接受者之间的耦合关系。将这个(处理者)对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理他为止。

     2.UML

      

 

【理解】

      1)场景:1.业务流程非常复杂,2.通常一个请求者,多个处理者,处理过程有顺序。

      2)通俗来讲,职责链模式,就是把处理者重新做了一层封装,因为处理对象的有序性需求,Handler的实现往往有2个必要的方法,其一是对处理者的节点顺序处理,当前节点处理完成之后使其能够到下一个节点处理。其二是处理当前节点所需要的业务逻辑。

 

【Demo】

      根据现实生活中“考驾照”的需求,写了一个小demo,规则:

      1.考驾照只能按照顺序来,考完一门考下一门。

      2.没门考试通过生成随机数来判别考试是否通过,大于60分则通过,小于则继续从该节点继续考试。

      1)uml展示:

         

 

     2) 代码展示:

     1.流程抽象类

package chainFcar;
/*** 模拟处理考试流程* 规则:* 1.考生只有依次考过了:科目一、科目二、科目三、科目四,方可拿到驾照* 2.各次考试考过与否成绩均存档* 3.中途未通过考试,下次从上次考试节点继续考试* * @author zhenhua.zhang*/
public abstract class AbsExamHandler {/*** 公共方法,每个节点都会执行,子类不用Override*/protected void commonMethod(){//todo 公共方法:eg ->//1.通过id查询考生信息//2.核对考生准考证、身份证等信息//3.and so on  这块可以拆分多个protected方法}/*** 不同流程的业务起点*/public abstract void handle(Object o, VincentChain chain);/*** 不同流程下的业务规则* @return*/public abstract Boolean businessCheck();protected void logBeforeExam(Integer type){System.out.println("Vincent你的考试科目" + type + "的考试马上开始!");}protected void logDuringExam(Integer type,Double score){System.out.println("Vincent你的考试科目" + type + "的成绩是:" + score + "分");}protected void logPassExam(Integer type){if(type == 4){System.out.println("Vincent,你通过了科目" + type + "的考核,恭喜你拿到了驾驶证!");return;}System.out.println("Vincent,你通过了科目" + type + "的考核,接下来好好准备下一门科目" + (type+1) + "的考核吧!");}protected void logFailExam(Integer type){System.out.println("Vincent,你未通过了科目" + type + "的考核,请再接再厉!!!");}}

       2.科目一~四实现类

package chainFcar;/*** 科目一考试* @author zhenhua.zhang**/
public class ProjectOneExamHandler extends AbsExamHandler{@Overridepublic void handle(Object o, VincentChain chain){//todoif(businessCheck()){chain.doChain(o);}}@Overridepublic Boolean businessCheck(){logBeforeExam(1);//生成1~100之间随机数,代表考试成绩,成绩大于60,可以考下一门考试Double score = (Double)(Math.random()*100);logDuringExam(1,score);if(score >= 60){logPassExam(1);return true;}logFailExam(1);return false;}
}
package chainFcar;
/*** 科目二考试* @author zhenhua.zhang**/
public class ProjectTwoExamHandler extends AbsExamHandler{@Overridepublic void handle(Object o, VincentChain chain){//todoif(businessCheck()){chain.doChain(o);}}@Overridepublic Boolean businessCheck(){logBeforeExam(2);//生成1~100之间随机数,代表考试成绩,成绩大于60,可以考下一门考试Double score = (Double)(Math.random()*100);logDuringExam(2,score);if(score >= 60){logPassExam(2);return true;}logFailExam(2);return false;}
}
package chainFcar;/** 科目一考试* @author zhenhua.zhang*/
public class ProjectThreeExamHandler extends AbsExamHandler{@Overridepublic void handle(Object o, VincentChain chain){//todoif(businessCheck()){chain.doChain(o);}}@Overridepublic Boolean businessCheck(){logBeforeExam(3);//生成1~100之间随机数,代表考试成绩,成绩大于60,可以考下一门考试Double score = (Double)(Math.random()*100);logDuringExam(3,score);if(score >= 60){logPassExam(3);return true;}logFailExam(3);return false;}
}
package chainFcar;/*** 科目四考试* @author zhenhua.zhang**/
public class ProjectFourExamHandler extends AbsExamHandler{@Overridepublic void handle(Object o, VincentChain chain){//todo otherif(businessCheck()){chain.doChain(o);}}@Overridepublic Boolean businessCheck(){logBeforeExam(4);Double score = (Double)(Math.random()*100);logDuringExam(4,score);if(score >= 60){logPassExam(4);return true;}logFailExam(4);return false;}
}

    3.职责链分发、流程管理类

package chainFcar;import java.util.ArrayList;
import java.util.List;/*** 职责链分发,流程管理* @author zhenhua.zhang**/
public class VincentChain {//处理流程handler集合,利用List有序特性List<AbsExamHandler> handler = new ArrayList<AbsExamHandler>();//index标识位,用于索引handlerint index = 0;/*** 按顺序向handler中添加元素* @param absHandler* @return*/public void addHandler(AbsExamHandler absHandler){handler.add(absHandler);//return this;}/*** 调用AbsExamHandler流程*/public void doChain(Object o){if(handler.size() == index){return;}AbsExamHandler handlerNew = handler.get(index);index++;handlerNew.handle(o,this);}
}

4.客户端启动访问类

package chainFcar;/*** * @author zhenhua.zhang**/public class VincentClient {public static void main(String[] args){VincentChain chain = new VincentChain();ProjectOneExamHandler one = new ProjectOneExamHandler();ProjectTwoExamHandler two = new ProjectTwoExamHandler();ProjectThreeExamHandler three = new ProjectThreeExamHandler();ProjectFourExamHandler four = new ProjectFourExamHandler();chain.addHandler(one);chain.addHandler(two);chain.addHandler(three);chain.addHandler(four);Object o = new Object();chain.doChain(o);}}

    执行结果:

    

      如图,因为考科目二的时候,随机数生成的成绩只有43.72分,所以到该节点便终止了考试流程。

 

【小结】

      一.上述例子已经展示了职责链模式的使用,关键点在于有序,这里用到了List<AbsExamHandler>的有序性保证顺序执行。

      二.优缺点比较:

      优点:

      1.长流程保证代码整洁、清晰,流程节点的顺序能仍能很好保证。

      缺点:

      1.(处理者)在实际的处理中,如果跳过了,就并没有发挥任何的作用。那么当这个链结构比较长,比较复杂的话,会产生很多的内存垃圾对象。这也就是职责链的最大缺点之所在。

       2.可能不容易观察运行时的特征,有碍于除错。

      三、对比链表

      1. 链表是一个链状结构,每个节点有一个next属性去指向他的下一节点。
      2. 链表有一个Header节点,然后用户每次必须通过头节点,然后去遍历寻找每一个节点。
      可见,职责链相比链表,每次走流程,无需从头节点开始。

    

      That's all.

这篇关于【Design Pattern】长流程 - 职责链模式的运用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1109660

相关文章

Security OAuth2 单点登录流程

单点登录(英语:Single sign-on,缩写为 SSO),又译为单一签入,一种对于许多相互关连,但是又是各自独立的软件系统,提供访问控制的属性。当拥有这项属性时,当用户登录时,就可以获取所有系统的访问权限,不用对每个单一系统都逐一登录。这项功能通常是以轻型目录访问协议(LDAP)来实现,在服务器上会将用户信息存储到LDAP数据库中。相同的,单一注销(single sign-off)就是指

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

poj 2431 poj 3253 优先队列的运用

poj 2431: 题意: 一条路起点为0, 终点为l。 卡车初始时在0点,并且有p升油,假设油箱无限大。 给n个加油站,每个加油站距离终点 l 距离为 x[i],可以加的油量为fuel[i]。 问最少加几次油可以到达终点,若不能到达,输出-1。 解析: 《挑战程序设计竞赛》: “在卡车开往终点的途中,只有在加油站才可以加油。但是,如果认为“在到达加油站i时,就获得了一

模版方法模式template method

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/template-method 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤。 上层接口有默认实现的方法和子类需要自己实现的方法

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

迭代器模式iterator

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/iterator 不暴露集合底层表现形式 (列表、 栈和树等) 的情况下遍历集合中所有的元素

《x86汇编语言:从实模式到保护模式》视频来了

《x86汇编语言:从实模式到保护模式》视频来了 很多朋友留言,说我的专栏《x86汇编语言:从实模式到保护模式》写得很详细,还有的朋友希望我能写得更细,最好是覆盖全书的所有章节。 毕竟我不是作者,只有作者的解读才是最权威的。 当初我学习这本书的时候,只能靠自己摸索,网上搜不到什么好资源。 如果你正在学这本书或者汇编语言,那你有福气了。 本书作者李忠老师,以此书为蓝本,录制了全套视频。 试

kubelet组件的启动流程源码分析

概述 摘要: 本文将总结kubelet的作用以及原理,在有一定基础认识的前提下,通过阅读kubelet源码,对kubelet组件的启动流程进行分析。 正文 kubelet的作用 这里对kubelet的作用做一个简单总结。 节点管理 节点的注册 节点状态更新 容器管理(pod生命周期管理) 监听apiserver的容器事件 容器的创建、删除(CRI) 容器的网络的创建与删除

利用命令模式构建高效的手游后端架构

在现代手游开发中,后端架构的设计对于支持高并发、快速迭代和复杂游戏逻辑至关重要。命令模式作为一种行为设计模式,可以有效地解耦请求的发起者与接收者,提升系统的可维护性和扩展性。本文将深入探讨如何利用命令模式构建一个强大且灵活的手游后端架构。 1. 命令模式的概念与优势 命令模式通过将请求封装为对象,使得请求的发起者和接收者之间的耦合度降低。这种模式的主要优势包括: 解耦请求发起者与处理者