设计模式系列-责任链模式

190 阅读4分钟

责任链模式

责任链模式会把将处理者对象连成一条链,并在该链上传递请求,执行链上有多个节点,每个节点都有机会(条件匹配)处理请求事务,如果某个节点处理完了就可以根据实际业务需求传递给下一个节点继续处理或者返回处理完毕。

优点

  • 简化了对象,链中的处理者并不知道整条链的结构,他们只需要保持一个对指向链中下一任处理对象的引用就好
  • 降低了耦合度,将请求的发送者和接受者解耦,请求者无需知道到底是哪个对象处理了,只管提交请求就好了。
  • 增强了给对象委派责任的灵活性,可以随意调整处理顺序
  • 能够很容易的动态增加删除新的请求处理类

缺点

  • 不一定保证请求一定被接收
  • 性能可能比较低,有的请求可能是从整条链的头部遍历到尾部
  • 调试起来比较麻烦

使用场景

  • 有多个对象可以处理同一个请求,具体哪个对象处理该请求由运行时刻自动确定
  • 不明确指定接收者的情况下,向多个对象中的一个提交一个请求
  • 可动态指定一组对象处理请求

重要组成成员

  • AbstractHandler:抽象处理者
  • Handler:具体处理者
  • AbstractRequest:抽象请求者
  • Request:具体处理者
  • Client:组织请求与处理顺序

示例--请假审批

今天,小明来找小组长请假,说他家里出了点状况,需要请一个月的假,可这么长的假期小组长是没权利做主的,小组长签完字,就叫他去找部门经理,然后小明拿着请假条去找部门经理,部门经理签字了,又让小明去找总经理签字,然后小明又拿着请假条去找总经理,最后总经理签字,这个假才算请完。

这样的话,其中涉及到的类如图所示

请假条抽象

public interface ILeave {
    String getName();//请假人姓名
    int getNum();//请假天数
    String getContent();//请假条内容
}

请假条

public class Leave implements ILeave{
    private String name;//姓名
    private int num;//请假天数
    private String content;//请假内容

    public Leave(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 = 0;
    private int numEnd = 0;

    //领导上面还有领导
    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;
    }

    //提交请假条
    public final void submit(ILeave leave){
        if(0 == this.numStart){
            return;
        }

        //如果请假天数达到该领导者的处理要求
        if(leave.getNum() >= this.numStart){
            this.handleLeave(leave);

            //如果还有上级 并且请假天数超过了当前领导的处理范围
            if(null != this.nextHandler && leave.getNum() > numEnd){
                this.nextHandler.submit(leave);//继续提交
            }
        }
    }

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

小组长

public class GroupLeader extends Handler {

    public GroupLeader() {
        //小组长处理1-3天的请假
        super(Handler.NUM_ONE, Handler.NUM_THREE);
    }

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

部门经理

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

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

总经理

public class BigManager extends Handler {
    public BigManager() {
        //部门经理处理7天以上的请假
        super(Handler.NUM_SEVEN);
    }

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

客户端

public static void main(String[] args) {
        //请假条来一张
        ILeave leave = new Leave("小花",5,"身体不适");

        //各位领导
        Handler groupLeader = new GroupLeader();
        Handler manager = new Manager();
        Handler bigManager = new BigManager();

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

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

应用

  • serverlet中的Filter

  • Dubbo中的Filter

  • Mybatiss中的Plugin

参考网址

责任链模式

责任链模式

责任链模式实现的三种方式