Spring框架之责任链模式 (Chain of Responsibility Pattern)

责任链模式(Chain of Responsibility Pattern)详解

责任链模式(Chain of Responsibility Pattern)是一种行为型设计模式,旨在将请求的发送者和处理者解耦。它通过创建一条 处理请求的责任链,使得多个对象都有机会处理请求,从而避免了请求的发送者和处理者之间的紧耦合。责任链模式的核心思想是将请求沿着链传递,直到有一个对象处理它为止。

在责任链模式中,处理请求的对象(处理者)被组织成一个链条,链中的每个对象都负责对请求进行处理。如果当前对象不能处理请求,它将把请求传递给链中的下一个对象,直到请求被处理为止。

1. 责任链模式的定义

1.1 什么是责任链模式?

责任链模式通过将一系列的处理对象组成一条链,使得每个处理对象都有机会去处理请求。每个处理对象都包含指向下一个处理对象的引用,如果当前处理对象无法处理请求,它会将请求传递给链中的下一个对象,直到请求被某个处理对象处理为止。

1.2 责任链模式的关键思想
  • 解耦请求发送者和请求处理者:请求的发送者并不直接知道具体由哪个处理者来处理请求,而是通过一条链来传递请求。
  • 动态地调整请求的处理顺序:可以灵活地改变责任链中处理者的顺序,甚至可以动态地添加新的处理者。
  • 每个处理者只关心自己能处理的请求:如果某个处理者无法处理请求,它将把请求传递给链中的下一个处理者。

2. 责任链模式的结构

责任链模式通常由以下角色构成:

  1. Handler(处理者):定义了一个接口或抽象类,声明了处理请求的方法,并持有对下一个处理者的引用。
  2. ConcreteHandler(具体处理者):实现了 Handler 接口,并定义了如何处理请求。如果当前处理者不能处理请求,它会将请求传递给链中的下一个处理者。
  3. Client(客户端):负责创建链并将请求传递给责任链的第一个处理者。
类图
    +------------------+        
    |    Handler       |        
    +------------------+        
    | + handleRequest()|        
    | + setNextHandler()|        
    +------------------+        
           ^                    
           |                    
    +------------------------+   
    | ConcreteHandlerA       |  
    +------------------------+  
    | + handleRequest()      |  
    | + setNextHandler()     |  
    +------------------------+   
           ^                    
           |                    
    +------------------------+   
    | ConcreteHandlerB       |  
    +------------------------+  
    | + handleRequest()      |  
    | + setNextHandler()     |  
    +------------------------+   
           ^                    
           |                    
    +------------------------+   
    | ConcreteHandlerC       |  
    +------------------------+  
    | + handleRequest()      |  
    | + setNextHandler()     |  
    +------------------------+  

3. 责任链模式的实现

通过一个实际的例子来演示责任链模式的实现。假设我们有一个处理请求的系统,每个请求有一个不同的级别(如:低、中、高),我们希望根据请求的级别来选择处理它的处理者。

3.1 Java 示例代码
// 抽象处理者
abstract class Handler {
    protected Handler nextHandler;

    public void setNextHandler(Handler nextHandler) {
        this.nextHandler = nextHandler;
    }

    public abstract void handleRequest(int level);
}

// 具体处理者A
class ConcreteHandlerA extends Handler {
    @Override
    public void handleRequest(int level) {
        if (level <= 10) {
            System.out.println("ConcreteHandlerA处理请求,级别:" + level);
        } else if (nextHandler != null) {
            nextHandler.handleRequest(level); // 转发请求
        }
    }
}

// 具体处理者B
class ConcreteHandlerB extends Handler {
    @Override
    public void handleRequest(int level) {
        if (level <= 20) {
            System.out.println("ConcreteHandlerB处理请求,级别:" + level);
        } else if (nextHandler != null) {
            nextHandler.handleRequest(level); // 转发请求
        }
    }
}

// 具体处理者C
class ConcreteHandlerC extends Handler {
    @Override
    public void handleRequest(int level) {
        if (level <= 30) {
            System.out.println("ConcreteHandlerC处理请求,级别:" + level);
        } else if (nextHandler != null) {
            nextHandler.handleRequest(level); // 转发请求
        }
    }
}

// 客户端代码
public class ChainOfResponsibilityPatternDemo {
    public static void main(String[] args) {
        // 创建责任链中的处理者
        Handler handlerA = new ConcreteHandlerA();
        Handler handlerB = new ConcreteHandlerB();
        Handler handlerC = new ConcreteHandlerC();

        // 设置责任链
        handlerA.setNextHandler(handlerB);
        handlerB.setNextHandler(handlerC);

        // 客户端发出请求
        System.out.println("请求级别为10:");
        handlerA.handleRequest(10); // 由A处理

        System.out.println("\n请求级别为15:");
        handlerA.handleRequest(15); // 由B处理

        System.out.println("\n请求级别为25:");
        handlerA.handleRequest(25); // 由C处理

        System.out.println("\n请求级别为35:");
        handlerA.handleRequest(35); // 没有处理者能处理该请求
    }
}

输出结果

请求级别为10:
ConcreteHandlerA处理请求,级别:10

请求级别为15:
ConcreteHandlerB处理请求,级别:15

请求级别为25:
ConcreteHandlerC处理请求,级别:25

请求级别为35:
(无输出)
3.2 解释
  • ConcreteHandlerA 只能处理请求级别小于或等于 10 的请求。如果请求级别大于 10,它会将请求转发给 ConcreteHandlerB 处理。
  • ConcreteHandlerB 只能处理请求级别小于或等于 20 的请求,超过这个级别的请求会被转发给 ConcreteHandlerC
  • ConcreteHandlerC 只能处理请求级别小于或等于 30 的请求。
  • 如果请求级别超过 30,责任链中没有任何处理者可以处理此请求,系统将不会输出任何内容。

4. 责任链模式的应用场景

责任链模式适用于以下场景:

  1. 多个对象可以处理一个请求:当一个请求可以由多个对象中的任何一个对象来处理时,可以使用责任链模式来处理。例如:请求的处理依赖于请求的级别或类型。
  2. 请求处理者的顺序不确定:当请求处理的顺序可以灵活变化时,可以使用责任链模式。例如:日志记录系统中的不同日志级别处理器,日志处理器可以按优先级顺序链式连接。
  3. 请求的处理者不一定知道最终的处理者是谁:责任链模式通过将请求沿链传递,避免了请求发送者和处理者之间的紧耦合。
  4. 动态地为请求分配处理者:通过责任链模式,可以动态地改变请求的处理顺序,甚至添加新的处理者。

5. 责任链模式的优缺点

5.1 优点
  • 降低耦合度:请求的发送者不需要知道哪个具体对象会处理请求,只需要知道责任链的起始处理者即可。这使得发送者和处理者之间的耦合度较低。
  • 增强灵活性:通过改变责任链的顺序,可以灵活地调整请求处理的方式,而无需修改请求的发送者或处理者。
  • 可扩展性强:责任链模式可以轻松地扩展新的处理者,只需将新的处理者添加到链中即可。
5.2 缺点
  • 可能导致请求得不到处理:如果责任链中的每个处理者都不能处理请求,那么请求就会一直在链中传递,直到超出链的范围。此时,客户端就无法获得处理结果。
  • 调试难度较大:由于请求会沿着链传递,可能会导致请求的跟踪变得较为困难,尤其是在责任链很长的情况下。

6. 责任链模式的实际应用

责任链模式广泛应用于实际项目中,特别是在处理一系列操作或任务时,以下是一些常见的应用场景:

  1. 日志处理系统:不同的日志记录器根据日志级别(如 DEBUG、INFO、ERROR)进行处理,日志处理器通常是责任链模式的一个典型应用。
  2. 审批流程:例如在企业的工作流系统中,某些任务可能需要多级审批,责任链模式可以非常自然地实现不同级别的审批处理。
  3. 事件处理系统:在 GUI 事件处理中,事件可以在多个事件监听器之间传递,直到某个监听器处理完该事件为止。
  4. 命令处理系统:例如,HTTP 请求处理中的路由系统,可以将请求传递给不同的处理器,根据请求的类型选择合适的处理逻辑。

7. 总结

责任链模式是一种行为型设计模式,通过将请求沿链传递,直到有一个对象处理它为止。它的主要优点是降低了请求发送者与处理者之间的耦合度,增强了系统的灵活性和可扩展性。然而,责任链模式也存在请求无法得到处理的风险以及调试上的难度。在适当的场景下,责任链模式可以有效地提高系统的设计灵活性和可维护性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/915116.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

重学 Android 自定义 View 系列(六):环形进度条

目标 自定义一个环形进度条&#xff0c;可以自定义其最大值、当前进度、背景色、进度色&#xff0c;宽度等信息。 最终效果如下&#xff08;GIF展示纯色有点问题&#xff09;&#xff1a; 1. 结构分析 背景圆环&#xff1a;表示进度条的背景。进度圆环&#xff1a;表示当前…

⚙️ 如何调整重试策略以适应不同的业务需求?

调整 Kafka 生产者和消费者的重试策略以适应不同的业务需求&#xff0c;需要根据业务的特性和容错要求来进行细致的配置。以下是一些关键的调整策略&#xff1a; 业务重要性&#xff1a; 对于关键业务消息&#xff0c;可以增加重试次数&#xff0c;并设置较长的重试间隔&#x…

总结拓展十五:特殊采购业务——寄售采购

1、寄售采购的定义 寄售采购是指供应商提供物料&#xff0c;并将它们存储在你处&#xff0c;在贵公司将这些物料从寄售库存提取&#xff08;转自有&#xff09;之前&#xff0c;该供应商一直是这些物料法律上的所有者。只有当这些物料被贵司转自有领用后&#xff0c;供应商才会…

RK3568平台开发系列讲解(GPIO篇)GPIO的sysfs调试手段

🚀返回专栏总目录 文章目录 一、内核配置二、GPIO sysfs节点介绍三、命令行控制GPIO3.1、sd导出GPIO3.2、设置GPIO方向3.3、GPIO输入电平读取3.4、GPIO输出电平设置四、Linux 应用控制GPIO4.1、控制输出4.2、输入检测4.3、使用 GPIO 中断沉淀、分享、成长,让自己和他人都能有…

【算法】——二分查找合集

阿华代码&#xff0c;不是逆风&#xff0c;就是我疯 你们的点赞收藏是我前进最大的动力&#xff01;&#xff01; 希望本文内容能够帮助到你&#xff01;&#xff01; 目录 零&#xff1a;二分查找工具 1&#xff1a;最基础模版 2&#xff1a;mid落点问题 一&#xff1a;最…

JAVA学习日记(十五) 数据结构

一、数据结构概述 数据结构是计算机底层存储、组织数据的方式。 数据结构是指数据相互之间以什么方式排列在一起的。 数据结构是为了更加方便的管理和使用数据&#xff0c;需要结合具体的业务场景来进行选择。 二、常见的数据结构 &#xff08;一&#xff09;栈 特点&…

Windows快速部署并使用GitHub上Swift项目

1.科学上网 2.找到项目&#xff0c;release部分&#xff0c;下载最新版的ZIP文件&#xff0c;并且打开&#xff0c;解压。 3.打开cmd&#xff0c;使用你做项目用的虚拟环境&#xff0c;安装必须安装的包文件 pip install ms-swift[llm] -U 类似这样子唰唰唰一堆安装好之后&am…

C++ | Leetcode C++题解之第552题学生出勤记录II

题目&#xff1a; 题解&#xff1a; class Solution { public:static constexpr int MOD 1000000007;vector<vector<long>> pow(vector<vector<long>> mat, int n) {vector<vector<long>> ret {{1, 0, 0, 0, 0, 0}};while (n > 0) {…

【精读】Kinodynamic Trajectory Optimization and Control for Car-Like Robots

原来阅读这个板块是我用来写小说灵感和摘抄笔记的&#xff0c;但是CSDN总说我重复率太高&#xff0c;mad以后改用来精读论文了 每天都在写不同的文章&#xff01;为什么&#xff1f;主要还是自我的研究进度跟不上课题组的进度 先给自己点根蜡烛11.15就开组会了我还没读完 ho…

学Linux的第八天

目录 管理进程 概念 程序、进程、线程 进程分类 查看进程 ps命令 unix 风格 bsd风格 GNU风格 top命令 格式 统计信息区 进程信息区&#xff1a;显示了每个进程的运行状态 kill命令 作用 格式 管理进程 概念 程序、进程、线程 程序&#xff1a; 二进制文件&…

uniCloud云对象调用第三方接口,根据IP获取用户归属地的免费API接口,亲测可用

需求 在2022年5月初&#xff0c;网络上各大平台上&#xff0c;都开始展示用户IP属地&#xff0c;在某音、某手等小视频平台以及各主流网站应用中&#xff0c;都展示IP归属地&#xff0c;如下图所示&#xff1a; 解决办法 收费文档的肯定有很多&#xff0c;基本你百度搜“归…

Leetcode - 143双周赛

目录 一&#xff0c;3345. 最小可整除数位乘积 I 二&#xff0c;3346. 执行操作后元素的最高频率 I 1.差分数组 2.同向三指针 滑动窗口 三&#xff0c; 3348. 最小可整除数位乘积 II 一&#xff0c;3345. 最小可整除数位乘积 I 本题直接暴力枚举&#xff0c;题目求 >n…

VS2022项目配置笔记

文章目录 $(ProjectDir&#xff09;与 $(SolutionDir) 宏附加包含目录VC目录和C/C的区别 $(ProjectDir&#xff09;与 $(SolutionDir) 宏 假设有一个解决方案 MySolution&#xff0c;其中包含两个项目 ProjectA 和 ProjectB&#xff0c;目录结构如下&#xff1a; C:\Projects\…

ReactPress:深入解析技术方案设计与源码

ReactPress Github项目地址&#xff1a;https://github.com/fecommunity/reactpress 欢迎提出宝贵的建议&#xff0c;欢迎一起共建&#xff0c;感谢Star。 ReactPress是一个基于React框架开发的开源发布平台&#xff0c;它不仅仅是一个简单的博客系统&#xff0c;更是一个功能全…

[编译报错]ImportError: No module named _sqlite3解决办法

1. 问题描述&#xff1a; 在使用python进行代码编译时&#xff0c;提示下面报错&#xff1a; "/home/bspuser/BaseTools/Source/Python/Workspace/WorkspaceDatabase.py", line 18, in <module>import sqlite3File "/usr/local/lib/python2.7/sqlite3/_…

信号量和线程池

1.信号量 POSIX信号量&#xff0c;用与同步操作&#xff0c;达到无冲突的访问共享资源目的&#xff0c;POSIX信号量可以用于线程间同步 初始化信号量 #include <semaphore.h> int sem_init(sem_t *sem, int pshared, unsigned int value); sem&#xff1a;指向sem_t类…

泷羽sec学习打卡-Linux基础2

声明 学习视频来自B站UP主 泷羽sec,如涉及侵权马上删除文章 笔记的只是方便各位师傅学习知识,以下网站只涉及学习内容,其他的都与本人无关,切莫逾越法律红线,否则后果自负 关于Linux的那些事儿-Base2 一、Linux-Base2linux有哪些目录呢&#xff1f;不同目录下有哪些具体的文件呢…

【Android、IOS、Flutter、鸿蒙、ReactNative 】约束布局

Android XML 约束布局 参考 TextView居中 TextView 垂直居中并且靠右 TextView 宽高设置百分比 宽和高的比例 app:layout_constraintDimensionRatio"h,2:1" 表示子视图的宽高比为2:1&#xff0c;其中 h表示保持宽度不变&#xff0c;高度自动调整。 最大宽度 设…

使用HTML、CSS和JavaScript创建动态圣诞树

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 ✨特色专栏&#xff1a…

golang分布式缓存项目 Day1 LRU 缓存淘汰策略

注&#xff1a;该项目原作者&#xff1a;https://geektutu.com/post/geecache-day1.html。本文旨在记录本人做该项目时的一些疑惑解答以及部分的测试样例以便于本人复习。 LRU缓存淘汰策略 三种缓存淘汰策略 FIFO&#xff08;First In, First Out&#xff09;先进先出 原理&…