当前位置: 首页 > article >正文

责任链设计模式详解

责任链设计模式详解

一、定义

责任链设计模式(Chain of Responsibility Pattern)是一种行为设计模式,它允许多个对象有机会处理请求,从而避免请求的发送者和接收者之间的耦合。这种模式将这些对象连接成一条链,并沿着这条链传递请求,直到有一个对象处理它为止。 就是说每个结点会处理一件事情,如果结点间出现异常,那么链路就会中断。

二、责任链的主要结构

主要结构

责任链模式的核心结构可以分为以下几个部分:

  1. 抽象处理者(Handler)
    • 定义一个处理请求的接口(通常是一个抽象类),包含一个方法来处理请求,以及一个指向下一个处理者的引用。
    • 提供设置和获取下一个处理者的方法。
  2. 具体处理者(ConcreteHandler)
    • 继承或实现抽象处理者接口。
    • 具体处理请求的实现,如果当前处理者不能处理该请求,则将其传递给下一个处理者。
  3. 客户端(Client)
    • 创建处理链的实例并将请求传递给链中的第一个处理者。

三、工作原理

当一个请求发送到责任链中的第一个处理者时,它会检查是否能够处理该请求。如果能够处理,处理者就会处理请求;如果不能处理,它就会将请求传递给下一个处理者,直到找到能够处理请求的处理者或者链的末端。

四、优缺点

优点
  • 解耦:请求的发送者和接收者之间没有直接关系,发送者无需知道具体哪个处理者处理了请求。
  • 灵活性:可以通过动态地添加或删除责任链中的处理者来改变处理请求的逻辑。
  • 增强代码的可扩展性:可以在不修改现有代码的情况下增加新的处理者。
缺点
  • 性能问题:如果链条太长,可能会导致性能问题,尤其是在每个处理者只做很少的处理时。
  • 调试复杂性:由于请求在多个处理者之间传递,调试起来可能会比较困难。

五、Spring MVC框架的责任链模式应用

springmvc流程:

在这里插入图片描述

  1. 用户发起请求,请求先被 Servlet 拦截转发给 Spring MVC 框架
  2. Spring MVC 里面的 DispatcherSerlvet 核心控制器,会接收到请求并转发给HandlerMapping
  3. HandlerMapping 负责解析请求,根据请求信息和配置信息找到匹配的 Controller类,不过这里如果有配置拦截器,就会按照顺序执行拦截器里面的 preHandle方法
  4. 找到匹配的 Controller 以后,把请求参数传递给 Controller 里面的方法
  5. Controller 中的方法执行完以后,会返回一个 ModeAndView,这里面会包括视图名称和需要传递给视图的模型数据
  6. 视图解析器根据名称找到视图,然后把数据模型填充到视图里面再渲染成 Html 内容返回给客户端
过滤器链(Filter Chain):
  • 过滤器是基于 Servlet 规范的,它们用于在请求到达目标 Servlet 之前以及响应返回客户端之前执行一些通用的任务。过滤器链是基于 Servlet 容器的,通常用于处理跨请求的通用任务,比如日志记录、安全检查、压缩等。
  • 责任链模式体现在多个过滤器按配置的顺序依次执行,如果某个过滤器决定拦截请求,则可以中止链条的继续传递。
  • 过滤器链的执行流程
    • 客户端发出请求。
    • 请求首先到达过滤器链,按照配置顺序依次执行。
    • 如果某个过滤器决定不放行请求,可以直接返回响应,从而中止后续的处理流程。
    • 如果所有过滤器都放行,则请求继续到达 Spring MVC 的 DispatcherServlet。
拦截器链(Handler Interceptor):
  • 拦截器是 Spring MVC 提供的更加精细的请求处理机制,允许在控制器方法之前和之后进行拦截处理。
  • HandlerInterceptor 接口提供了三个主要方法:preHandlepostHandleafterCompletion,这些方法可以在请求处理的不同阶段被调用。
  • 在 Spring MVC 的配置中,多个拦截器可以被配置成链式调用,它们之间的关系就是一种典型的责任链模式。请求到达 DispatcherServlet 时,首先会通过拦截器链的 preHandle 方法,如果所有的拦截器都返回 true,请求才会继续到达处理器(Handler)。处理完成后,依次调用拦截器的 postHandleafterCompletion 方法。
  • 拦截器链的执行流程
  • 请求到达 DispatcherServlet 后,根据 Handler Mapping 找到对应的 Controller 处理器。
  • 在 Controller 方法执行之前,拦截器链中的 preHandle 方法依次执行。如果所有 preHandle 方法都返回 true,则继续执行 Controller 方法。
  • Controller 方法执行完毕后,拦截器链中的 postHandle 方法按照配置顺序依次执行。
  • 最后,afterCompletion 方法会在请求完成后执行,用于进行一些资源清理或日志记录等操作。
处理器映射链(Handler Mapping Chain):
  • 在 Spring MVC 中,处理器映射(Handler Mapping)用于将请求 URL 映射到相应的处理器(Controller)。
  • 可以配置多个处理器映射,通过责任链模式来依次检查每一个映射器,直到找到一个合适的处理器。
  • 处理器映射链的执行流程
  • DispatcherServlet 根据配置的多个 Handler Mapping 依次查找合适的 Controller。
  • 当找到一个匹配的 Controller 时,停止继续查找,并调用该 Controller 处理请求。

六、责任链示例

abstract class Handler {
    protected Handler next;

    public void setNext(Handler next) {
        this.next = next;
    }

    public abstract void handleRequest(int request);
}

class ConcreteHandler1 extends Handler {
    @Override
    public void handleRequest(int request) {
        if (request < 10) {
            System.out.println("Handler1处理请求: " + request);
        } else if (next != null) {
            next.handleRequest(request);
        }
    }
}

class ConcreteHandler2 extends Handler {
    @Override
    public void handleRequest(int request) {
        if (request >= 10 && request < 20) {
            System.out.println("Handler2处理请求: " + request);
        } else if (next != null) {
            next.handleRequest(request);
        }
    }
}

class Client {
    public static void main(String[] args) {
        Handler handler1 = new ConcreteHandler1();
        Handler handler2 = new ConcreteHandler2();

        handler1.setNext(handler2);

        int[] requests = {5, 14, 22};

        for (int request : requests) {
            handler1.handleRequest(request);
        }
    }
}


http://www.kler.cn/a/289242.html

相关文章:

  • 〔 MySQL 〕数据类型
  • JVM 中的完整 GC 流程
  • Ceph 中Crush 算法的理解
  • 建筑施工特种作业人员安全生产知识试题
  • 【大数据测试HBase数据库 — 详细教程(含实例与监控调优)】
  • C++模板特化实战:在使用开源库boost::geometry::index::rtree时,用特化来让其支持自己的数据类型
  • B端系统门门清之:SCRM系统,为客户管理加上社交,如虎添翼
  • MySQL事务处理(TransAction)
  • 苹果11月推出新款M4 Mac:Mac mini设计焕新 MacBook Pro仅例行更新
  • wsl下将Ubuntu从c盘移动到其他盘
  • Linux——IO模型_多路转接(epoll)
  • 监控平台之针对vue,react上报
  • ​yum安装/更新时报错:SyntaxError: invalid syntax​
  • 全局网络代理的使用与选择
  • css的position定位的属性
  • 【Java那些事】关于Git的使用
  • Datawhale X 李宏毅苹果书 AI夏令营|机器学习基础之案例学习
  • Vue -- 总结 02
  • adb大全指令(持续更新)
  • 动态住宅IP代理的搭建指南:实现高效网络访问
  • 数据访问:JPA关联MyBatis
  • Elasticsearch的Restful风格API
  • 达梦常用SQL及脚本工具
  • 哈希 详解
  • echart自适应tree树图,结构组织图模板
  • 国赛数模C题模型(五)