责任链模式
1、定义:
- 将能够处理同一类请求的对象连成一条链,所提交的请求沿着链传递,链上的对象逐个判断是否有能力处理该请求 ,如果能则处理; 如果不能则传递给链上的下一个对象
2、场景:
- 大学中奖学金审批
- 公司公文审批、请假流程审批
3、开发中常见的场景
- java中的异常机制就是一种责任链模式。一个try可以对应多个catch,当第一个catch不匹配类型时,自动跳到下一个catch;
- servlet开发中,过滤器的链式调用;
- struts中,拦截器的调用也是典型的责任链模式;
4、 添加处理对象
- 由于责任链的创建完全在客户端,因此新增的具体处理者对原有类库没有任何影响,只需要添加新的类,然后再客户端调用时添加即可,符合开闭原则。
案例:
我们可以在请假流程中,增加新的“副总经理”角色,审批大于等于10小于20天的请假
审批流程变为:
1、如果天数小于3天,主任审批
2、天数大于等于3天,小于10天经理审批
3、10<=天数<20,副总经理审批
4、20<=天数<30,总经理审批
5、天数>=30,拒绝
5、优点
- 1、请求的发送者和接收者解耦
- 2、可以控制执行顺序
- 3、符合开闭原则和单一职责原则
核心代码如下:
/**
* 抽象类
* @author tqq
* @date 2021-04-02-14:36
*/
public abstract class Leader {
protected String name;
protected Leader nextLeader;//责任链的后继对象
public Leader(String name) {
this.name = name;
}
//责任链的后继对象
public void setNextLeader(Leader nextLeader) {
this.nextLeader = nextLeader;
}
/**
* 处理请求的核心业务方法
* @param request
*/
public abstract void handleRequest(LeaveRequest request);
}
package chainofpesp;
/**
* 封装请假的基本信息
* @author tqq
* @date 2021-04-02-14:34
*/
public class LeaveRequest {
private String empName;
private Integer leaveDays;
private String reason;
/**
*省略get/set方法 构造方法
**/
}
/**
* 经理
* @author tqq
* @date 2021-04-02-14:39
*/
public class Manager extends Leader{
final private Integer TEN_DAYS = 3;
public Manager(String name) {
super(name);
}
@Override
public void handleRequest(LeaveRequest request) {
if (request.getLeaveDays()<TEN_DAYS){
System.out.println("员工:"+request.getEmpName()+"请假,天数:"+request.getLeaveDays()+",理由:"+request.getReason());
System.out.println("经理:"+this.name+",审批通过!");
}
else {
if (this.nextLeader!=null){
this.nextLeader.handleRequest(request);
}
}
}
}
/**
* @author tqq
* @date 2021-04-02-14:55
*/
public class Client {
public static void main(String[] args) {
Director a = new Director("张三");
Manager b = new Manager("李四");
GeneralManager c = new GeneralManager("王五");
ViceGeneralManager d = new ViceGeneralManager("刘二");
//组织责任链对象的关系
a.setNextLeader(b);
b.setNextLeader(d);
d.setNextLeader(c);
//开始请假
LeaveRequest request = new LeaveRequest("tqq", 10, "回学校做毕设");
a.handleRequest(request);
}
}