责任链设计模式详解

2024-08-29 13:52
文章标签 设计模式 详解 责任

本文主要是介绍责任链设计模式详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

责任链设计模式详解

一、定义

责任链设计模式(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 {@Overridepublic void handleRequest(int request) {if (request < 10) {System.out.println("Handler1处理请求: " + request);} else if (next != null) {next.handleRequest(request);}}
}class ConcreteHandler2 extends Handler {@Overridepublic 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.chinasem.cn/article/1118046

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

K8S(Kubernetes)开源的容器编排平台安装步骤详解

K8S(Kubernetes)是一个开源的容器编排平台,用于自动化部署、扩展和管理容器化应用程序。以下是K8S容器编排平台的安装步骤、使用方式及特点的概述: 安装步骤: 安装Docker:K8S需要基于Docker来运行容器化应用程序。首先要在所有节点上安装Docker引擎。 安装Kubernetes Master:在集群中选择一台主机作为Master节点,安装K8S的控制平面组件,如AP

嵌入式Openharmony系统构建与启动详解

大家好,今天主要给大家分享一下,如何构建Openharmony子系统以及系统的启动过程分解。 第一:OpenHarmony系统构建      首先熟悉一下,构建系统是一种自动化处理工具的集合,通过将源代码文件进行一系列处理,最终生成和用户可以使用的目标文件。这里的目标文件包括静态链接库文件、动态链接库文件、可执行文件、脚本文件、配置文件等。      我们在编写hellowor

LabVIEW FIFO详解

在LabVIEW的FPGA开发中,FIFO(先入先出队列)是常用的数据传输机制。通过配置FIFO的属性,工程师可以在FPGA和主机之间,或不同FPGA VIs之间进行高效的数据传输。根据具体需求,FIFO有多种类型与实现方式,包括目标范围内FIFO(Target-Scoped)、DMA FIFO以及点对点流(Peer-to-Peer)。 FIFO类型 **目标范围FIFO(Target-Sc

019、JOptionPane类的常用静态方法详解

目录 JOptionPane类的常用静态方法详解 1. showInputDialog()方法 1.1基本用法 1.2带有默认值的输入框 1.3带有选项的输入对话框 1.4自定义图标的输入对话框 2. showConfirmDialog()方法 2.1基本用法 2.2自定义按钮和图标 2.3带有自定义组件的确认对话框 3. showMessageDialog()方法 3.1

脏页的标记方式详解

脏页的标记方式 一、引言 在数据库系统中,脏页是指那些被修改过但还未写入磁盘的数据页。为了有效地管理这些脏页并确保数据的一致性,数据库需要对脏页进行标记。了解脏页的标记方式对于理解数据库的内部工作机制和优化性能至关重要。 二、脏页产生的过程 当数据库中的数据被修改时,这些修改首先会在内存中的缓冲池(Buffer Pool)中进行。例如,执行一条 UPDATE 语句修改了某一行数据,对应的缓

OmniGlue论文详解(特征匹配)

OmniGlue论文详解(特征匹配) 摘要1. 引言2. 相关工作2.1. 广义局部特征匹配2.2. 稀疏可学习匹配2.3. 半稠密可学习匹配2.4. 与其他图像表示匹配 3. OmniGlue3.1. 模型概述3.2. OmniGlue 细节3.2.1. 特征提取3.2.2. 利用DINOv2构建图形。3.2.3. 信息传播与新的指导3.2.4. 匹配层和损失函数3.2.5. 与Super