Skip to content

责任链模式

ZHI-XINHUA edited this page Jan 11, 2019 · 1 revision

责任链模式(chain of responsibility)

在责任链模式里,很多对象由每一个对象对其下家的引用而连接起来形成一条链。请求在这个链上传递,直到链上的某一个对象决定处理此请求。发出这个请求的客户端并不知道链上的那个对象最终处理这个请求,这使得系统可以在不影响考核单的情况下动态地重新组织链和分配责任。

1、角色

  • 抽象处理者(Handler)角色:定义一个处理请求的接口/抽象类,如果需要,可以定义出一个方法,以设定和返回对下家的应用;抽象方法handleRequest()规范子类处理请求的操作;
  • 具体处理者(ConcreteHandler)角色:具体处理者接收到请求后,可选择将请求处理掉,或者将请求传递给下家。

2、demo场景

公司里面,请假条的审批过程: • 如果请假天数小于3天,主任审批 • 如果请假天数大于等于3天,小于10天,经理审批 • 如果大于等于10天,小于30天,总经理审批 • 如果大于等于30天,提示拒绝

urm:

(1)创建请假条基本信息

/**
 * 请假条基本信息
 */
public class LeaveRequest {
    /**员工名称**/
    private String empName;

    /**请假天数**/
    private int leaveDays;

    /**请假原因**/
    private String reason;

    public LeaveRequest(String empName, int leaveDays, String reason) {
        this.empName = empName;
        this.leaveDays = leaveDays;
        this.reason = reason;
    }

    public String getEmpName() {
        return empName;
    }

    public void setEmpName(String empName) {
        this.empName = empName;
    }

    public int getLeaveDays() {
        return leaveDays;
    }

    public void setLeaveDays(int leaveDays) {
        this.leaveDays = leaveDays;
    }

    public String getReason() {
        return reason;
    }

    public void setReason(String reason) {
        this.reason = reason;
    }

    @Override
    public String toString() {
        return "员工:" + empName + ",请假"+leaveDays+"天,理由是:"+reason;
    }
}

(2)创建领导抽象类 Leader

/**
 * 领导抽象类
 */
public abstract class Leader {
    /**名字**/
    protected String name;

    /**直属领导**/
    protected Leader nextLeader;

    public Leader(String name) {
        this.name = name;
    }

    /**
     * 责任链上的后继对象
     * @param nextLeader
     */
    public void setNextLeader(Leader nextLeader){
        this.nextLeader = nextLeader;
    }

    /**
     * 处理请求的核心业务方法
     * @param request
     */
    public abstract void handleRequest(LeaveRequest request);


}

(3)创建主任审批、经理、总经理

/**
 * 主任
 */
public class Director extends Leader {
    public Director(String name) {
        super(name);
    }

    @Override
    public void handleRequest(LeaveRequest request) {
        if(request.getLeaveDays()<3){
            request.toString();
            System.out.println("主任:"+name+" 审批通过!");
        }else {
            if(nextLeader!=null){
                this.nextLeader.handleRequest(request);//传递下一位领导
            }
        }
    }
}
/**
 * 经理
 */
public class Manager extends Leader {
    public Manager(String name) {
        super(name);
    }

    @Override
    public void handleRequest(LeaveRequest request) {
        if(request.getLeaveDays()<10){
            request.toString();
            System.out.println("经理:"+name+" 审批通过!");
        }else {
            if(nextLeader!=null){
                this.nextLeader.handleRequest(request);//传递下一位领导
            }
        }
    }
}
/**
 * 总经理
 */
public class GeneralManager extends Leader {
    public GeneralManager(String name) {
        super(name);
    }

    @Override
    public void handleRequest(LeaveRequest request) {
        if(request.getLeaveDays()<30){
            request.toString();
            System.out.println("总经理:"+name+" 审批通过!");
        }else {
            System.out.println(request.getEmpName()+"请假"+request.getLeaveDays()+"天这么多,是不是不想干了?");
        }
    }
}

(4)测试

public class Client {
    public static void main(String[] args) {
        Director a = new Director("张三");
        Manager b = new Manager("李四");
        GeneralManager c = new GeneralManager("王五");

        //设置组织关系
        a.setNextLeader(b);
        b.setNextLeader(c);

        //开始请假
        LeaveRequest leaveRequest = new LeaveRequest("james",3,"世界这么大,想去看看");
        a.handleRequest(leaveRequest);
    }
}

经理:李四 审批通过!

3、开发中常见的场景

  • Java中,异常机制就是一种责任链模式。一个try可以对应多个catch, 当第一个catch不匹配类型,则自动跳到第二个catch.
  • Javascript语言中,事件的冒泡和捕获机制。Java语言中,事件的处理采用观察者模式。
  • Servlet开发中,过滤器的链式处理
  • Struts2中,拦截器的调用也是典型的责任链模式