设计模式_行为型模式 -《责任链模式》
创始人
2024-05-14 06:39:35
0

设计模式_行为型模式 -《责任链模式》

笔记整理自 黑马程序员Java设计模式详解, 23种Java设计模式(图解+框架源码分析+实战)

概述

在现实生活中,常常会出现这样的事例:一个请求有多个对象可以处理,但每个对象的处理条件或权限不同。例如,公司员工请假,可批假的领导有部门负责人、副总经理、总经理等,但每个领导能批准的天数不同,员工必须根据自己要请假的天数去找不同的领导签名,也就是说员工必须记住每个领导的姓名、电话和地址等信息,这增加了难度。这样的例子还有很多,如找领导出差报销、生活中的“击鼓传花”游戏等。

定义

  • 责任链模式,又名职责链模式,为了避免请求发送者与多个请求处理者耦合在一起,将所有请求的处理者通过前一对象记住其下一个对象的引用而连成一条链;当有请求发生时,可将请求沿着这条链传递,直到有对象处理它为止。

image-20230109124843959

结构

责任链模式 (Chain of Responsibility Pattern) 主要包含以下角色:

  • 抽象处理者(Handler)角色:定义一个处理请求的接口,包含抽象处理方法和一个后继连接(下一个对象的引用)。
  • 具体处理者(Concrete Handler)角色:实现抽象处理者的处理方法,判断能否处理本次请求,如果可以处理请求则处理,否则将该请求转给它的后继者。
  • 客户类(Client)角色:创建处理链,并向链头的具体处理者对象提交请求,它不关心处理细节和请求的传递过程。

案例实现

现需要开发一个请假流程控制系统。请假 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天,身体不适。
    总经理审批:同意
    流程结束!
    

优缺点

优点

  • 降低了对象之间的耦合度
    • 该模式降低了请求发送者和接收者的耦合度。
  • 增强了系统的可扩展性
    • 可以根据需要增加新的请求处理类,满足开闭原则。
  • 增强了给对象指派职责的灵活性
    • 当工作流程发生变化,可以动态地改变链内的成员或者修改它们的次序,也可动态地新增或者删除责任。
  • 责任链简化了对象之间的连接
    • 一个对象只需保持一个指向其后继者的引用,不需保持其他所有处理者的引用,这避免了使用众多的 if 或者 if···else 语句。
  • 责任分担
    • 每个类只需要处理自己该处理的工作,不能处理的传递给下一个对象完成,明确各类的责任范围,符合类的单一职责原则。

缺点

  • 不能保证每个请求一定被处理。由于一个请求没有明确的接收者,所以不能保证它一定会被处理,该请求可能一直传到链的末端都得不到处理。
  • 对比较长的责任链,请求的处理可能涉及多个处理对象,系统性能将受到一定影响。
  • 责任链建立的合理性要靠客户端来保证,增加了客户端的复杂性,可能会由于责任链的错误设置而导致系统出错,如可能会造成循环调用。

源码解析-JavaWeb

在 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 后置处理
    

相关内容

热门资讯

【NI Multisim 14...   目录 序言 一、工具栏 🍊1.“标准”工具栏 🍊 2.视图工具...
银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...
不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
AWSECS:访问外部网络时出... 如果您在AWS ECS中部署了应用程序,并且该应用程序需要访问外部网络,但是无法正常访问,可能是因为...
Android|无法访问或保存... 这个问题可能是由于权限设置不正确导致的。您需要在应用程序清单文件中添加以下代码来请求适当的权限:此外...
北信源内网安全管理卸载 北信源内网安全管理是一款网络安全管理软件,主要用于保护内网安全。在日常使用过程中,卸载该软件是一种常...
AWSElasticBeans... 在Dockerfile中手动配置nginx反向代理。例如,在Dockerfile中添加以下代码:FR...
AsusVivobook无法开... 首先,我们可以尝试重置BIOS(Basic Input/Output System)来解决这个问题。...
ASM贪吃蛇游戏-解决错误的问... 要解决ASM贪吃蛇游戏中的错误问题,你可以按照以下步骤进行:首先,确定错误的具体表现和问题所在。在贪...
月入8000+的steam搬砖... 大家好,我是阿阳 今天要给大家介绍的是 steam 游戏搬砖项目,目前...