学习设计模式之责任链模式

Posted 南淮北安

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了学习设计模式之责任链模式相关的知识,希望对你有一定的参考价值。

文章目录

一、定义


击鼓传雷,看上图你是否想起周星驰有一个电影,大家坐在海边围成一个圈,拿着一个点燃的炸弹,互相传递。

责任链模式的核心是解决一组服务中的先后执行处理关系,就有点像你没钱花了,需要家庭财务支出审批,10块钱以下找闺女审批,100块钱先闺女审批在媳妇审批。你可以理解想象成当你要跳槽的时候被安排的明明白白的被各个领导签字放行

二、问题背景


在本案例中我们模拟在618大促期间的业务系统上线审批流程场景

像是这些一线电商类的互联网公司,阿里、京东、拼多多等,在618期间都会做一些运营活动场景以及提供的扩容备战,就像过年期间百度的红包一样。但是所有开发的这些系统都需要陆续的上线,因为临近618有时候也有一些紧急的调整的需要上线,但为了保障线上系统的稳定性是尽可能的减少上线的,也会相应的增强审批力度。就像一级响应、二级响应一样。

而这审批的过程在随着特定时间点会增加不同级别的负责人加入,每个人就像责任链模式中的每一个核心点。

对于研发小伙伴并不需要关心具体的审批流程处理细节,只需要知道这个上线更严格,级别也更高,但对于研发人员来说同样是点击相同的提审按钮,等待审核。

接下来我们就模拟这样一个业务诉求场景,使用责任链的设计模式来实现此功能

(1)场景模拟工程

itstack-demo-design-13-00
└── src
    └── main
        └── java
            └── org.itstack.demo.design
                └── AuthService.java

这里的代码结构比较简单,只有一个模拟审核和查询审核结果的服务类。相当于你可以调用这个类去审核工程和获取审核结构,这部分结果信息是模拟的写到缓存实现

(2)模拟审核服务

public class AuthService 

    private static Map<String, Date> authMap = new ConcurrentHashMap<String, Date>();

    public static Date queryAuthInfo(String uId, String orderId) 
        return authMap.get(uId.concat(orderId));
    

    public static void auth(String uId, String orderId) 
        authMap.put(uId.concat(orderId), new Date());
    


这里面提供了两个接口一个是查询审核结果(queryAuthInfo)、另外一个是处理审核(auth)。

这部分是把由谁审核的和审核的单子ID作为唯一key值记录到内存Map结构中

三、违背设计模式的设计实现

这里我们先使用最直接的方式来实现功能

按照我们的需求审批流程,平常系统上线只需要三级负责人审批就可以,但是到了618大促时间点,就需要由二级负责以及一级负责人一起加入审批系统上线流程。

在这里我们使用非常直接的if判断方式来实现这样的需求

1. 工程结构

itstack-demo-design-13-01
└── src
    └── main
        └── java
            └── org.itstack.demo.design
                └── AuthController.java

这部分非常简单的只包含了一个审核的控制类,就像有些伙伴开始写代码一样,一个类写所有需求

2. 代码实现

public class AuthController 

    private SimpleDateFormat f = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");// 时间格式化

    public AuthInfo doAuth(String uId, String orderId, Date authDate) throws ParseException 

        // 三级审批
        Date date = AuthService.queryAuthInfo("1000013", orderId);
        if (null == date) return new AuthInfo("0001", "单号:", orderId, " 状态:待三级审批负责人 ", "王工");

        // 二级审批
        if (authDate.after(f.parse("2020-06-01 00:00:00")) && authDate.before(f.parse("2020-06-25 23:59:59"))) 
            date = AuthService.queryAuthInfo("1000012", orderId);
            if (null == date) return new AuthInfo("0001", "单号:", orderId, " 状态:待二级审批负责人 ", "张经理");
        

        // 一级审批
        if (authDate.after(f.parse("2020-06-11 00:00:00")) && authDate.before(f.parse("2020-06-20 23:59:59"))) 
            date = AuthService.queryAuthInfo("1000011", orderId);
            if (null == date) return new AuthInfo("0001", "单号:", orderId, " 状态:待一级审批负责人 ", "段总");
        

        return new AuthInfo("0001", "单号:", orderId, " 状态:审批完成");
    

这里从上到下分别判断了在指定时间范围内由不同的人员进行审批,就像618上线的时候需要三个负责人都审批才能让系统进行上线。

像是这样的功能看起来很简单的,但是实际的业务中会有很多部门,但如果这样实现就很难进行扩展,并且在改动扩展调整也非常麻烦

四、问题改进

接下来使用装饰器模式来进行代码优化,也算是一次很小的重构。

责任链模式可以让各个服务模块更加清晰,而每一个模块间可以通过next的方式进行获取。

而每一个next是由继承的统一抽象类实现的。最终所有类的职责可以动态的进行编排使用,编排的过程可以做成可配置化

1. 工程结构

itstack-demo-design-13-02
└── src
    └── main
        └── java
            └── org.itstack.demo.design
                ├── impl
                │    ├── Level1AuthLink.java
                │    ├── Level2AuthLink.java
                │    └── Level3AuthLink.java
                ├── AuthInfo.java
                └── AuthLink.java

责任链模型结构:

上图是这个业务模型中责任链结构的核心部分,通过三个实现了统一抽象类AuthLink的不同规则,再进行责任编排模拟出一条链路。这个链路就是业务中的责任链。

一般在使用责任链时候如果是场景比较固定,可以通过写死到代码中进行初始化。但如果业务场景经常变化可以做成xml配置的方式进行处理,也可以落到库里进行初始化操作

2. 代码实现

(1)责任链中返回对象定义

public class AuthInfo 

    private String code;
    private String info = "";

    public AuthInfo(String code, String ...infos) 
        this.code = code;
        for (String str:infos)
            this.info = this.info.concat(str);
        
    
    
    // ...get/set


这个类的是包装了责任链处理过程中返回结果的类,方面处理每个责任链的返回信息。

(2)链路抽象类定义

public abstract class AuthLink 

    protected Logger logger = LoggerFactory.getLogger(AuthLink.class);

    protected SimpleDateFormat f = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");// 时间格式化
    protected String levelUserId;                           // 级别人员ID
    protected String levelUserName;                         // 级别人员姓名
    private AuthLink next;                                  // 责任链

    public AuthLink(String levelUserId, String levelUserName) 
        this.levelUserId = levelUserId;
        this.levelUserName = levelUserName;
    

    public AuthLink next() 
        return next;
    

    public AuthLink appendNext(AuthLink next) 
        this.next = next;
        return this;
    

    public abstract AuthInfo doAuth(String uId, String orderId, Date authDate);



这部分是责任链,链接起来的核心部分。AuthLink next,重点在于可以通过next方式获取下一个链路需要处理的节点。

levelUserId、levelUserName,是责任链中的公用信息,标记每一个审核节点的人员信息。
抽象类中定义了一个抽象方法,abstract AuthInfo doAuth,这是每一个实现者必须实现的类,不同的审核级别处理不同的业务

(3)三个审核实现类

Level1AuthLink:

public class Level1AuthLink extends AuthLink 

    public Level1AuthLink(String levelUserId, String levelUserName) 
        super(levelUserId, levelUserName);
    

    public AuthInfo doAuth(String uId, String orderId, Date authDate) 
        Date date = AuthService.queryAuthInfo(levelUserId, orderId);
        if (null == date) 
            return new AuthInfo("0001", "单号:", orderId, " 状态:待一级审批负责人 ", levelUserName);
        
        AuthLink next = super.next();
        if (null == next) 
            return new AuthInfo("0000", "单号:", orderId, " 状态:一级审批完成负责人", " 时间:", f.format(date), " 审批人:", levelUserName);
        

        return next.doAuth(uId, orderId, authDate);
    



Level2AuthLink:

public class Level2AuthLink extends AuthLink 

    private Date beginDate = f.parse("2020-06-11 00:00:00");
    private Date endDate = f.parse("2020-06-20 23:59:59");

    public Level2AuthLink(String levelUserId, String levelUserName) throws ParseException 
        super(levelUserId, levelUserName);
    

    public AuthInfo doAuth(String uId, String orderId, Date authDate) 
        Date date = AuthService.queryAuthInfo(levelUserId, orderId);
        if (null == date) 
            return new AuthInfo("0001", "单号:", orderId, " 状态:待二级审批负责人 ", levelUserName);
        
        AuthLink next = super.next();
        if (null == next) 
            return new AuthInfo("0000", "单号:", orderId, " 状态:二级审批完成负责人", " 时间:", f.format(date), " 审批人:", levelUserName);
        

        if (authDate.before(beginDate) || authDate.after(endDate)) 
            return new AuthInfo("0000", "单号:", orderId, " 状态:二级审批完成负责人", " 时间:", f.format(date), " 审批人:", levelUserName);
        

        return next.doAuth(uId, orderId, authDate);
    



Level3AuthLink:

public class Level3AuthLink extends AuthLink 

    private Date beginDate = f.parse("2020-06-01 00:00:00");
    private Date endDate = f.parse("2020-06-25 23:59:59");

    public Level3AuthLink(String levelUserId, String levelUserName) throws ParseException 
        super(levelUserId, levelUserName);
    

    public AuthInfo doAuth(String uId, String orderId, Date authDate) 
        Date date = AuthService.queryAuthInfo(levelUserId, orderId);
        if (null == date) 
            return new AuthInfo("0001", "单号:", orderId, " 状态:待三级审批负责人 ", levelUserName);
        
        AuthLink next = super.next();
        if (null == next) 
            return new AuthInfo("0000", "单号:", orderId, " 状态:三级审批负责人完成", " 时间:", f.format(date), " 审批人:", levelUserName);
        

        if (authDate.before(beginDate) || authDate.after(endDate)) 
            return new AuthInfo("0000", "单号:", orderId, " 状态:三级审批负责人完成", " 时间:", f.format(date), " 审批人:", levelUserName);
        

        return next.doAuth(uId, orderId, authDate);
    



如上三个类;Level1AuthLink、Level2AuthLink、Level3AuthLink,实现了不同的审核级别处理的简单逻辑。

例如第一个审核类中会先判断是否审核通过,如果没有审核通过则返回结果给调用方,引导去审核。(这里简单模拟审核后有时间信息不为空,作为判断条件)
判断完成后获取下一个审核节点;super.next();,如果不存在下一个节点,则直接返回结果。
之后是根据不同的业务时间段进行判断是否需要,二级和一级的审核。
最后返回下一个审核结果;next.doAuth(uId, orderId, authDate);,有点像递归调用

3. 测试验证

@Test
public void test_AuthLink() throws ParseException 
    AuthLink authLink = new Level3AuthLink("1000013", "王工")
            .appendNext(new Level2AuthLink("1000012", "张经理")
                    .appendNext(new Level1AuthLink("1000011", "段总")));

    logger.info("测试结果:", JSON.toJSONString(authLink.doAuth("小傅哥", "1000998004813441", new Date())));

    // 模拟三级负责人审批
    AuthService.auth("1000013", "1000998004813441");
    logger.info("测试结果:", "模拟三级负责人审批,王工");
    logger.info("测试结果:", JSON.toJSONString(authLink.doAuth("小傅哥", "1000998004813441", new Date())));

    // 模拟二级负责人审批
    AuthService.auth("1000012", "1000998004813441");
    logger.info("测试结果:", "模拟二级负责人审批,张经理");
    logger.info("测试结果:", JSON.toJSONString(authLink.doAuth("小傅哥", "1000998004813441", new Date())));

    // 模拟一级负责人审批
    AuthService.auth("1000011", "1000998004813441");
    logger.info("测试结果:", "模拟一级负责人审批,段总");
    logger.info("测试结果:", JSON.toJSONString(authLink.doAuth("小傅哥", "1000998004813441", new Date())));


这里包括最核心的责任链创建,实际的业务中会包装到控制层;

AuthLink authLink = new Level3AuthLink("1000013", "王工") .appendNext(new Level2AuthLink("1000012", "张经理") .appendNext(new Level1AuthLink("1000011", "段总"))); 

通过把不同的责任节点进行组装,构成一条完整业务的责任链。

接下里不断的执行查看审核链路authLink.doAuth(...),通过返回结果对数据进行3、2、1级负责人审核,直至最后审核全部完成。

五、总结

从上面代码从if语句重构到使用责任链模式开发可以看到,我们的代码结构变得清晰干净了,也解决了大量if语句的使用。并不是if语句不好,只不过if语句并不适合做系统流程设计,但是在做判断和行为逻辑处理中还是非常可以使用的。

在我们前面学习结构性模式中讲到过组合模式,它像是一颗组合树一样,我们搭建出一个流程决策树。其实这样的模式也是可以和责任链模型进行组合扩展使用,而这部分的重点在于如何关联链路的关联,最终的执行都是在执行在中间的关系链。

责任链模式很好的处理单一职责和开闭原则,简单了耦合也使对象关系更加清晰,而且外部的调用方并不需要关心责任链是如何进行处理的*(以上程序中可以把责任链的组合进行包装,在提供给外部使用)*。但除了这些优点外也需要是适当的场景才进行使用,避免造成性能以及编排混乱调试测试疏漏问题。

创作打卡挑战赛 赢取流量/现金/CSDN周边激励大奖

以上是关于学习设计模式之责任链模式的主要内容,如果未能解决你的问题,请参考以下文章

Java 设计模式之责任链学习与掌握

Java 设计模式之责任链学习与掌握

一天学习一个设计模式之责任链模式

Java设计模式之八 ----- 责任链模式和命令模式

学习设计模式之责任链模式

Java进阶篇设计模式之八 ----- 责任链模式和命令模式