• 【java设计模式】之 责任链(chain of resposibility)模式


    责任链模式,顾名思义,就是一条链。这个链到底是怎么运行的呢?它主要是将能够处理同一类请求的对象连成一条链,所提交的请求沿着链传递,链上的对象逐个判断是否有能力处理该请求,如果能则处理,如果不能则传递给链上的下一个对象进行处理,以此类推。 
      比如说公司请假需要审批,举个不恰当的例子,如果请假小于3天,主管审批;3-10天的,经理审批;10-30天的,总经理审批;超过30天的,不批准等等。这就得一步步去判断,如果撇开设计模式不看的话,那么我们可以使用if…else…把它解决了,但是问题可想而知,实际中的复杂程度时远远超过这个例子的。 
      下面使用责任链模式来实现一下这个场景,先来看一下它们之间的关系结构: 
    这里写图片描述 
      Leader是个抽象类,里面有个方法是获取下一个Leader,Director、Manager和GeneralManager分别继承这个抽象类。LeaveRequest中封装了请假的相关信息。下面来实现一下这个责任链模式。 
    首先是封装请假信息的类:

    /**
     * @Description 封装请假的基本信息
     * @author shanheyongmu
     *
     */
    public class LeaveRequest {
    
        private String name;
        private int leaveDays;
        private String reason;
    
        public LeaveRequest(String name, int leaveDays, String reason) {
            super();
            this.name = name;
            this.leaveDays = leaveDays;
            this.reason = reason;
        }
        //省略get和set方法
    }   

    然后就是定义Leader这个抽象类了

    /**
     * @Description 领导的抽象类
     * @author shanheyongmu
     *
     */
    public abstract class Leader {
    
        protected String name;
        protected Leader nextLeader; //责任链上的后继对象,即这个对象无法处理,就转移给下一个Leader
    
        public Leader(String name) {
            super();
            this.name = name;
        }
        // 设定责任链上的后继对象
        public void setNextLeader(Leader nextLeader) {
            this.nextLeader = nextLeader;
        }
    
        /**
         * 处理请求的核心的业务方法
         * 需要不同继承该类的领导自己实现
         */
        public abstract void handleRequest(LeaveRequest request);
    
    }

    主管、经理和总经理都需要继承该类,并实现处理请假的这个方法。

    /**
     * @Description 主任
     * @author shanheyongmu
     *
     */
    public class Director extends Leader {
    
        public Director(String name) {
            super(name);
        }
    
        @Override
        public void handleRequest(LeaveRequest request) {
    
            int days = request.getLeaveDays(); //获取请假天数
            String name = request.getName(); //获取请假人姓名
            String reason = request.getReason(); // 获取请假理由
    
            if(days <= 3) { //如果满足3天内的要求,主任直接审批
                System.out.println("员工" + name + "请假" + days + "天,理由:" + reason);
                System.out.println("主任" + this.name + "审批通过");
            } else {
                System.out.println("请假天数过多,主任" + this.name + "没法处理");
                if(this.nextLeader != null) { //否则,如果链上存在下一个Leader,就让他处理
                    this.nextLeader.handleRequest(request);
                } 
            }
        }
    
    }
    /**
     * @Description 经理
     * @author shanheyongmu
     *
     */
    public class Manager extends Leader {
    
        public Manager(String name) {
            super(name);
        }
    
        @Override
        public void handleRequest(LeaveRequest request) {
    
            int days = request.getLeaveDays(); //获取请假天数
            String name = request.getName(); //获取请假人姓名
            String reason = request.getReason(); // 获取请假理由
    
            if(days <= 10) { //如果满足10天内的要求,经理直接审批
                System.out.println("员工" + name + "请假" + days + "天,理由:" + reason);
                System.out.println("经理" + this.name + "审批通过");
            } else {
                System.out.println("请假天数过多,经理" + this.name + "没法处理");
                if(this.nextLeader != null) { //否则,如果链上存在下一个Leader,就让他处理
                    this.nextLeader.handleRequest(request);
                } 
            }
        }
    
    }
    /**
     * @Description 总经理
     * @author shanheyongmu
     *
     */
    public class GeneralManager extends Leader {
    
        public GeneralManager(String name) {
            super(name);
        }
    
        @Override
        public void handleRequest(LeaveRequest request) {
    
            int days = request.getLeaveDays(); //获取请假天数
            String name = request.getName(); //获取请假人姓名
            String reason = request.getReason(); // 获取请假理由
    
            if(days <= 30) { //如果满足30天内的要求,总经理直接审批
                System.out.println("员工" + name + "请假" + days + "天,理由:" + reason);
                System.out.println("总经理" + this.name + "审批通过");
            } else {
                System.out.println("请假天数过多,总经理" + this.name + "没法处理");
                if(this.nextLeader != null) { //否则,如果链上存在下一个Leader,就让他处理
                    this.nextLeader.handleRequest(request);
                } else {
                    System.out.println("请假不成功");
                }
            }
        }
    
    }

    由此可见,他们处理请求的条件是不同的,而且只要不是自己处理范围之内的,就会将请求传给链上的下一位来处理。接下来写个客户端模拟下:

    public class Client {
    
        public static void main(String[] args) {
    
            Leader director = new Director("张三");
            Leader manager = new Manager("李四");
            Leader gManager = new GeneralManager("王五");
    
            // 组织好责任链对象的关系
            director.setNextLeader(manager);
            manager.setNextLeader(gManager);
    
            // 开始请假操作
            LeaveRequest request = new LeaveRequest("山河君", 15, "度假");
            director.handleRequest(request);
        }
    
    }

    运行结果:

    请假天数过多 主任张三没法处理
    请假天数过多,经理李四没法处理
    员工山河君请假15天,理由: 度假
    总经理王五审批通过

      在实际中,设置这个链可以放到配置文件中,可以直接修改配置文件即可,这个责任链模式扩展性也很好,比如现在有个副经理,专门处理10-20天的请假,那么我只要新添加一个副经理类即可,然后在责任链上配置一下就行了,对其他的类没有影响。 
      责任链模式使用的地方很多,比如我们常见的try…catch…就是,可以有多个catch,一个个往下判断;再比如servlet里面的过滤器,就是一个责任链;再比如struts2中的拦截器,也是使用的责任链模式。 

  • 相关阅读:
    团队的展示以及规划
    基于上次数独基础优化,并添加GUI界面
    利用程序随机构造N个已解答的数独棋盘
    本学期高级软件工程课程的实践项目的自我目标
    本学期高级软件工程课程的实践项目的自我目标
    案列分析
    编程作业
    构建之法
    软工一
    小黄衫获奖感言
  • 原文地址:https://www.cnblogs.com/shanheyongmu/p/6378637.html
Copyright © 2020-2023  润新知