设计模式之责任链模式【行为型模式】

学习的最大理由是想摆脱平庸,早一天就多一份人生的精彩;迟一天就多一天平庸的困扰。各位小伙伴,如果您:
想系统/深入学习某技术知识点…
一个人摸索学习很难坚持,想组团高效学习…
想写博客但无从下手,急需写作干货注入能量…
热爱写作,愿意让自己成为更好的人…


前言

一、概述
二、结构
三、案例实现
四、优缺点
五、源码解析


一、概述

在现实生活中,常常会出现这样的事例:一个请求有多个对象可以处理,但每个对象的处理条件或权限不同。例如,公司员工请假,可批假的领导有部门负责人、副总经理、总经理等,但每个领导能批准的天数不同,员工必须根据自己要请假的天数去找不同的领导签名,也就是说员工必须记住每个领导的姓名、电话和地址等信息,这增加了难度。这样的例子还有很多,如找领导出差报销、生活中的“击鼓传花”游戏等。

定义:

又名职责链模式,为了避免请求发送者与多个请求处理者耦合在一起,将所有请求的处理者通过前一对象记住其下一个对象的引用而连成一条链;当有请求发生时,可将请求沿着这条链传递,直到有对象处理它为止。

二、结构

职责链模式主要包含以下角色:

  • 抽象处理者(Handler)角色:定义一个处理请求的接口,包含抽象处理方法和一个后继连接。
  • 具体处理者(Concrete Handler)角色:实现抽象处理者的处理方法,判断能否处理本次请求,如果可以处理请求则处理,否则将该请求转给它的后继者。
  • 客户类(Client)角色:创建处理链,并向链头的具体处理者对象提交请求,它不关心处理细节和请求的传递过程。

三、案例实现

现需要开发一个请假流程控制系统。请假一天以下的假只需要小组长同意即可;请假1天到3天的假还需要部门经理同意;请求3天到7天还需要总经理同意才行。

类图如下:
在这里插入图片描述
代码如下:

//请假条
public class LeaveRequest {
   
    private String name;//姓名
    private int num;//请假天数
    private String content;//请假内容

    public LeaveRequest(String name, int num, String content) {
   
        this.name = name;
        this.num = num;
        this.content = content;
    }

    public String getName() {
   
        return name;
    }

    public int getNum() {
   
        return num;
    }

    public String getContent() {
   
        return content;
    }
}


//处理者抽象类
public abstract class Handler {
   
    protected final static int NUM_ONE = 1;
    protected final static int NUM_THREE = 3;
    protected final static int NUM_SEVEN = 7;

    //该领导处理的请假天数区间
    private int numStart;
    private int numEnd;

    //领导上面还有领导,声明后继者
    private Handler nextHandler;

    //设置请假天数范围 上不封顶
    public Handler(int numStart) {
   
        this.numStart = numStart;
    }

    //设置请假天数范围
    public Handler(int numStart, int numEnd) {
   
        this.numStart = numStart;
        this.numEnd = numEnd;
    }

    //设置上级领导
    public void setNextHandler(Handler nextHandler){
   
        this.nextHandler = nextHandler;
    }

    //各级领导处理请假条方法
    protected abstract void handleLeave(LeaveRequest leave);

    //提交请假条
    public final void submit(LeaveRequest leave){
   
        //该领导进行审批
        this.handleLeave(leave);
        if(this.nextHandler !=null && leave.getNum() > this.numEnd){
   
            //提交给上级领导进行审批
            this.nextHandler.submit(leave);
        }else {
   
            System.out.println("流程结束");
        }

    }


}

//小组长
public class GroupLeader extends Handler {
   
    public GroupLeader() {
   
        //小组长处理1天以下的请假
        super(0, Handler.NUM_ONE);
    }

    @Override
    protected void handleLeave(LeaveRequest leave) {
   
        System.out.println(leave.getName() + "请假" + leave.getNum() + "天," + leave.getContent() + "。");
        System.out.println("小组长审批:同意。");
    }
}

//部门经理
public class Manager extends Handler {
   
    public Manager() {
   
        //部门经理处理1-3天的请假
        super(Handler.NUM_ONE, Handler.NUM_THREE);
    }

    @Override
    protected void handleLeave(LeaveRequest leave) {
   
        System.out.println(leave.getName() + "请假" + leave.getNum() + "天," + leave.getContent() + "。");
        System.out.println("部门经理审批:同意。");
    }
}

//总经理
public class GeneralManager extends Handler {
   
    public GeneralManager() {
   
        //部门经理处理3-7天以上的请假
        super(Handler.NUM_THREE,Handler.NUM_SEVEN);
    }

    @Override
    protected void handleLeave(LeaveRequest leave) {
   
        System.out.println(leave.getName() + "请假" + leave.getNum() + "天," + leave.getContent() + "。");
        System.out.println("总经理审批:同意。");
    }
}
//测试类
public class Client {
   
    public static void main(String[] args) {
   
        //请假条来一张
        LeaveRequest leave = new LeaveRequest("小时代",5,"身体不适");

        //各位领导
        GroupLeader groupLeader = new GroupLeader();
        Manager manager = new Manager();
        GeneralManager generalManager = new GeneralManager();

        groupLeader.setNextHandler(manager);//小组长的领导是部门经理
        manager.setNextHandler(generalManager);//部门经理的领导是总经理
        //之所以在这里设置上级领导,是因为可以根据实际需求来更改设置,如果实战中上级领导人都是固定的,则可以移到领导实现类中。

        //提交申请
        groupLeader.submit(leave);
    }
}

在这里插入图片描述

四、优缺点

1,优点:

  • 降低了对象之间的耦合度

    该模式降低了请求发送者和接收者的耦合度。

  • 增强了系统的可扩展性

    可以根据需要增加新的请求处理类,满足开闭原则。

  • 增强了给对象指派职责的灵活性

    当工作流程发生变化,可以动态地改变链内的成员或者修改它们的次序,也可动态地新增或者删除责任。

  • 责任链简化了对象之间的连接

    一个对象只需保持一个指向其后继者的引用,不需保持其他所有处理者的引用,这避免了使用众多的 if 或者 if···else 语句。

  • 责任分担

    每个类只需要处理自己该处理的工作,不能处理的传递给下一个对象完成,明确各类的责任范围,符合类的单一职责原则。

2,缺点:

  • 不能保证每个请求一定被处理。由于一个请求没有明确的接收者,所以不能保证它一定会被处理,该请求可能一直传到链的末端都得不到处理。
  • 对比较长的职责链,请求的处理可能涉及多个处理对象,系统性能将受到一定影响。
  • 职责链建立的合理性要靠客户端来保证,增加了客户端的复杂性,可能会由于职责链的错误设置而导致系统出错,如可能会造成循环调用。

五、源码解析

在javaWeb应用开发中,FilterChain是职责链(过滤器)模式的典型应用,以下是Filter的模拟实现分析:

  • 模拟web请求Request以及web响应Response

    public interface Request{
         
     
    }
    
    public interface Response{
         
     
    }
    
  • 模拟web过滤器Filter

     public interface Filter {
         
     	public void doFilter(Request req,Response res,FilterChain c);
     }
    
  • 模拟实现具体过滤器

    public class FirstFilter implements Filter {
         
        @Override
        public void doFilter(Request request, Response response, FilterChain chain) {
         
    
            System.out.println("过滤器1 前置处理");
    
            // 先执行所有request再倒序执行所有response
            chain.doFilter(request, response);
    
            System.out.println("过滤器1 后置处理");
        }
    }
    
    public class SecondFilter  implements Filter {
         
        @Override
        public void doFilter(Request request, Response response, FilterChain chain) {
         
    
            System.out.println("过滤器2 前置处理");
    
            // 先执行所有request再倒序执行所有response
            chain.doFilter(request, response);
    
            System.out.println("过滤器2 后置处理");
        }
    }
    
  • 模拟实现过滤器链FilterChain

    public class FilterChain {
         
    
        private List<Filter> filters = new ArrayList<Filter>();
    
        private int index = 0;
    
        // 链式调用
        public FilterChain addFilter(Filter filter) {
         
            this.filters.add(filter);
            return this;
        }
    
        public void doFilter(Request request, Response response) {
         
            if (index == filters.size()) {
         
                return;
            }
            Filter filter = filters.get(index);
            index++;
            filter.doFilter(request, response, this);
        }
    }
    
  • 测试类

    public class Client {
         
        public static void main(String[] args) {
         
            Request  req = null;
            Response res = null ;
    
            FilterChain filterChain = new FilterChain();
            filterChain.addFilter(new FirstFilter()).addFilter(new SecondFilter());
            filterChain.doFilter(req,res);
        }
    }
    

在这里插入图片描述


总结

以上就是设计模式之责任链模式【行为型模式】的相关知识点,希望对你有所帮助。
积跬步以至千里,积怠惰以至深渊。时代在这跟着你一起努力哦!

相关推荐

  1. 设计模式(017)行为责任模式

    2024-01-10 10:44:02       11 阅读
  2. 设计模式行为模式责任模式

    2024-01-10 10:44:02       29 阅读
  3. 设计模式-行为模式-责任模式

    2024-01-10 10:44:02       15 阅读
  4. GO设计模式——15、责任模式行为

    2024-01-10 10:44:02       35 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-01-10 10:44:02       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-01-10 10:44:02       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-01-10 10:44:02       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-01-10 10:44:02       20 阅读

热门阅读

  1. MYSQL的SWITCH语句和循环语句

    2024-01-10 10:44:02       30 阅读
  2. QT组件学习(2)

    2024-01-10 10:44:02       34 阅读
  3. Golang企业面试题

    2024-01-10 10:44:02       33 阅读
  4. Docker 使用DockerFile构建Nodejs服务镜像

    2024-01-10 10:44:02       40 阅读
  5. Arduino驱动ISD1820音频录放模块(声音传感器)

    2024-01-10 10:44:02       37 阅读
  6. 互动直播 之 隐藏音频连麦者头像

    2024-01-10 10:44:02       34 阅读
  7. Go语言的编译过程

    2024-01-10 10:44:02       35 阅读
  8. 双机调度算法

    2024-01-10 10:44:02       36 阅读
  9. 力扣-135.分发糖果

    2024-01-10 10:44:02       30 阅读
  10. 基于长短期神经网络lstm的电子密度预测

    2024-01-10 10:44:02       38 阅读