笔记整理自 黑马程序员Java设计模式详解, 23种Java设计模式(图解+框架源码分析+实战)
在现实生活中,常常会出现这样的事例:一个请求有多个对象可以处理,但每个对象的处理条件或权限不同。例如,公司员工请假,可批假的领导有部门负责人、副总经理、总经理等,但每个领导能批准的天数不同,员工必须根据自己要请假的天数去找不同的领导签名,也就是说员工必须记住每个领导的姓名、电话和地址等信息,这增加了难度。这样的例子还有很多,如找领导出差报销、生活中的“击鼓传花”游戏等。
定义
责任链模式 (Chain of Responsibility Pattern) 主要包含以下角色:
现需要开发一个请假流程控制系统。请假 1 天以下的假只需要小组长同意即可;请假 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;}// 提交请假条public final void submit(LeaveRequest leave){if (0 == this.numStart) {return;}// 如果请假天数达到该领导者的处理要求if (leave.getNum() >= this.numStart) {this.handleLeave(leave);// 如果还有上级 并且请假天数超过了当前领导的处理范围if (this.nextHandler != null && leave.getNum() > numEnd) {this.nextHandler.submit(leave); // 继续往上提交 责任链模式} else {System.out.println("流程结束");}}}// 各级领导处理请假条方法protected abstract void handleLeave(LeaveRequest leave);
}
具体处理者角色
// 具体处理者-小组长
public class GroupLeader extends Handler {public GroupLeader() {// 小组长处理1天以下的请假super(0, Handler.NUM_ONE);}@Overrideprotected 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);}@Overrideprotected 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);}@Overrideprotected 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);}
}
输出
张三请假5天,身体不适。
小组长审批:同意
张三请假5天,身体不适。
部门经理审批:同意
张三请假5天,身体不适。
总经理审批:同意
流程结束!
优点
缺点
在 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 {@Overridepublic void doFilter(Request request, Response response, FilterChain chain) {System.out.println("过滤器1 前置处理");// 先执行所有request再倒序执行所有responsechain.doFilter(request, response);System.out.println("过滤器1 后置处理");}
}public class SecondFilter implements Filter {@Overridepublic void doFilter(Request request, Response response, FilterChain chain) {System.out.println("过滤器2 前置处理");// 先执行所有request再倒序执行所有responsechain.doFilter(request, response);System.out.println("过滤器2 后置处理");}
}
模拟实现过滤器链 FilterChain
public class FilterChain {private List filters = new ArrayList();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 前置处理
过滤器2 前置处理
过滤器2 后置处理
过滤器1 后置处理
下一篇:Redis 主从复制