责任链模式的理解和实践

        责任链模式(Chain of Responsibility)是行为型设计模式之一,它通过将多个对象连成一条链,并沿着这条链传递请求,直到有对象处理它为止。这个模式的主要目的是将请求的发送者和接收者解耦,使请求沿着处理链传递,直到被某个对象处理。本文将详细介绍责任链模式的理解和实践,并提供Java示例代码。

一、责任链模式的理解

定义与背景

        责任链模式是一种行为设计模式,它允许你将请求沿着处理者链进行传递。每个处理者对象都对请求进行某种形式的处理,然后决定是将请求传递给链中的下一个处理者对象,还是直接处理该请求。这种模式使得你可以在不明确指定请求处理对象的情况下,向多个对象发送请求。

主要角色

  • 抽象处理者(Handler):定义一个处理请求的接口,并包含一个后继者(successor)的引用,可以持有链中的下一个处理者对象。
  • 具体处理者(Concrete Handler):实现抽象处理者的接口,处理它所负责的请求。如果无法处理,则传递给后继者。
  • 客户端(Client):构建处理链,并向链中的第一个处理者发送请求。

工作原理

        责任链模式通过维护一个处理者对象的链表,使得每个对象都有机会处理请求。当一个请求被发送到链中的第一个处理者对象时,该对象会检查它是否能处理该请求。如果可以,则处理;如果不可以,则将该请求传递给链中的下一个处理者对象。这个过程一直持续到请求被某个对象处理为止,或者链的末端。

二、责任链模式的实践

        接下来,我们将通过Java代码来实现责任链模式。

  1. 定义抽象处理者

    首先,我们定义一个抽象处理者接口,该接口包含一个处理请求的方法和一个设置后继者的方法。

    public abstract class Handler {
        protected Handler successor;
    
        public void setSuccessor(Handler successor) {
            this.successor = successor;
        }
    
        public abstract void handleRequest(String request);
    }
  2. 实现具体处理者

    然后,我们实现几个具体处理者类,这些类将处理不同类型的请求。

    public class ConcreteHandler1 extends Handler {
        @Override
        public void handleRequest(String request) {
            if (request.equals("Handler1")) {
                System.out.println("ConcreteHandler1 handled the request: " + request);
            } else {
                if (successor != null) {
                    successor.handleRequest(request);
                }
            }
        }
    }
    
    public class ConcreteHandler2 extends Handler {
        @Override
        public void handleRequest(String request) {
            if (request.equals("Handler2")) {
                System.out.println("ConcreteHandler2 handled the request: " + request);
            } else {
                if (successor != null) {
                    successor.handleRequest(request);
                }
            }
        }
    }
    
    public class ConcreteHandler3 extends Handler {
        @Override
        public void handleRequest(String request) {
            if (request.equals("Handler3")) {
                System.out.println("ConcreteHandler3 handled the request: " + request);
            } else {
                // This handler is the last in the chain, no successor to pass the request to
            }
        }
    }

  3. 构建处理链并发送请求

    最后,我们在客户端代码中构建处理链,并向链中的第一个处理者发送请求。

    public class Client {
        public static void main(String[] args) {
            Handler handler1 = new ConcreteHandler1();
            Handler handler2 = new ConcreteHandler2();
            Handler handler3 = new ConcreteHandler3();
    
            handler1.setSuccessor(handler2);
            handler2.setSuccessor(handler3);
    
            // Test requests
            handler1.handleRequest("Handler1");
            handler1.handleRequest("Handler2");
            handler1.handleRequest("Handler3");
            handler1.handleRequest("Unknown");
        }
    }

  4. 运行结果

    运行客户端代码,你会看到以下输出:

    ConcreteHandler1 handled the request: Handler1
    ConcreteHandler2 handled the request: Handler2
    ConcreteHandler3 handled the request: Handler3

    对于未知请求("Unknown"),由于 ConcreteHandler3 是链中的最后一个处理者,并且它没有后继者,所以该请求不会被进一步处理。

三、责任链模式的优点与缺点

  1. 优点

    • 降低了耦合度:请求发送者和接收者之间解耦,发送者不需要知道接收者的具体实现。
    • 增强了灵活性:通过动态地改变链中的处理者,可以很容易地改变请求的处理流程。
    • 责任明确:每个处理者对象都明确知道它所负责处理的请求类型。
  2. 缺点

    • 性能问题:请求需要沿着链传递,直到被处理,这可能会增加系统的响应时间。
    • 调试困难:由于请求可能经过多个处理者对象,定位问题可能会变得复杂。
    • 链的构造:构建和维护处理链可能会比较复杂,特别是在处理者对象较多时。

四、应用场景

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

  • 事件处理系统:如GUI框架中的事件处理机制,事件沿着事件处理链传递,直到被某个处理器处理。
  • 审批流程:如请假审批流程,请求沿着审批链传递,直到被某个审批人批准或拒绝。
  • 过滤器机制:如Web应用中的过滤器链,请求和响应都沿着过滤器链传递,每个过滤器可以对请求和响应进行某种处理。

总结

        责任链模式通过将多个处理者对象连成一条链,并沿着这条链传递请求,直到请求被某个对象处理,从而实现了请求发送者和接收者之间的解耦。这种模式提高了系统的灵活性和可扩展性,但也带来了性能问题和调试困难。在实际应用中,我们需要根据具体场景和需求,权衡这些优缺点,决定是否使用责任链模式。

        通过本文的介绍和示例代码,相信你对责任链模式有了更深入的理解。希望这些知识和经验能够帮助你在实际开发中更好地应用该模式,设计出更加灵活和可维护的系统。

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

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

相关文章

软件工程——期末复习(3)

一、题目类(老师重点提到过的题目) 1、高可靠性是否意味着高可用性?试举例证明自己的观点? 答:高可靠性不意味着高可用性 可靠性说明系统已经准备好,马上可以使用;可用性是系统可以无故障的持续运行,是一…

SList(单链表)

文章目录 一:线性表二:数组2.1数组在内存中的存储 三:链式结构四:单链表4.1概念与结构4.1.1概念4.1.2 结构(节点)4.1.3链表的性质4.1.4链表的打印 4.2实现单链表 结语 欢迎大家来到我的博客,给生…

VTK知识学习(21)- 数据的读写

1、前言 对于应用程序而言,都需要处理特定的数据,VTK应用程序也不例外。 VTK应用程序所需的数据可以通过两种途径获取: 第一种是生成模型,然后处理这些模型数据(如由类 vtkCylinderSource 生成的多边形数据); 第二种是从外部存储介质里导…

Nignx部署Java服务测试使用的Spring Boot项目Demo

天行健,君子以自强不息;地势坤,君子以厚德载物。 每个人都有惰性,但不断学习是好好生活的根本,共勉! 文章均为学习整理笔记,分享记录为主,如有错误请指正,共同学习进步。…

虚幻引擎生存建造系统

先做一个建造预览模式,按下按键B后进入建造预览模式 首先创建自定义事件Preview Loop 用射线追踪摆放物体预览位置,并做一个预览材质 增强输入设置按键 每帧判断是否进入建造模式 预览模式制作成功! 接着做点击左键放置物品&#xff0…

Blender中使用BlenderGIS插件快速生成城市建筑模型

导入下载 BlenderGIS 插件 去github上下载其压缩包,地址如下: https://github.com/domlysz/BlenderGIS 在BlenderGIS中导入这个插件压缩包: 点击上方菜单栏的编辑,点击偏好设置 在插件>从磁盘安装中导入刚刚下载的压缩包 可…

C语言期末复习

1、任意输入一个半径给r&#xff0c;求圆的面积。 #include <stdio.h> #include <windows.h> void main() { double r,s; printf("输入一个半径给r"); scanf("%lf",&r); sr*r*3.1415926; printf("%lf",s); system(&qu…

深圳大学《2024年904自动控制原理真题》 (完整版)

本文内容&#xff0c;全部选自自动化考研联盟的&#xff1a;《深圳大学904自控考研资料》的真题篇。后续会持续更新更多学校&#xff0c;更多年份的真题&#xff0c;记得关注哦~ 目录 2024年真题 Part1&#xff1a;2024年完整版真题 2024年真题

transformers生成式对话机器人

简介 生成式对话机器人是一种先进的人工智能系统&#xff0c;它能够通过学习大量的自然语言数据来模拟人类进行开放、连贯且创造性的对话。与基于规则或检索式的聊天机器人不同&#xff0c;生成式对话机器人并不局限于预定义的回答集&#xff0c;而是可以根据对话上下文动态地…

NanoLog起步笔记-4-Server端的两个线程

nonolog起步笔记-4-Server端的两个线程 Server端的两个线程两个线程的角色与各自的职责RuntimeLogger::compressionThreadMain线程 详细学习一下相关的代码第三个线程第一次出现原位置swip buffer Server端的两个线程 如前所述&#xff0c;nanolog的server端&#xff0c;相对而…

Freertos任务切换

一、操作系统进行任务切换的时机&#xff1a; 采用信号量实现任务的互斥&#xff1a; 二、FreeRTOS 任务切换场合 PendSV 中断的时候提到了上下文(任务)切换被触发的场合&#xff1a; ● 可以执行一个系统调用 ● 系统滴答定时器(SysTick)中断。 1、执行系统调用 执行系统…

【硬件测试】基于FPGA的4FSK调制解调通信系统开发与硬件片内测试,包含信道模块,误码统计模块,可设置SNR

目录 1.算法仿真效果 2.算法涉及理论知识概要 3.Verilog核心程序 4.开发板使用说明和如何移植不同的开发板 5.完整算法代码文件获得 1.算法仿真效果 本文是之前写的文章: 《基于FPGA的4FSK调制解调系统,包含testbench,高斯信道模块,误码率统计模块,可以设置不同SNR》 的…

【Vue2+Element-ui】el-dialog宽度适配

1、不适配问题 分辨率100%-页面 分辨率150%-页面 在项目中&#xff0c;我开发分辨率一直是100%&#xff0c;但是客户使用的分辨率不相同&#xff0c;所以宽度要适配 2、解决-封装mixins.js 1)、封装的mixins 我将宽度设置成动态的&#xff0c;因为我的项目中需求不同。 expor…

Tr0ll: 1 Vulnhub靶机渗透笔记

Tr0ll: 1 本博客提供的所有信息仅供学习和研究目的&#xff0c;旨在提高读者的网络安全意识和技术能力。请在合法合规的前提下使用本文中提供的任何技术、方法或工具。如果您选择使用本博客中的任何信息进行非法活动&#xff0c;您将独自承担全部法律责任。本博客明确表示不支…

23. C++STL 9 (priority_queue的使用和适配实现详解)

⭐本篇重点&#xff1a; 1 priority_queue的使用与底层原理 2 使用容器来适配 priority_queue ⭐本篇代码&#xff1a;c学习 橘子真甜/c-learning-of-yzc - 码云 - 开源中国 (gitee.com) ⭐标⭐是比较重要的部分 目录 一. priority_queue&#xff08;优先级队列&#xff09;的…

十四、Pod的升级和回滚

当集群中的某个服务需要升级时,我们需要停止目前与该服务相关的所有Pod,然后下载新版本镜像并创建新的Pod。如果集群规模比较大,则这个工作变成了一个挑战,而且先全部停止然后逐步升级的方式会导致较长时间的服务不可用。Kubernetes提供了滚动升级功能来解决上述问题。 如…

中间件--MongoDB部署及初始化js脚本(docker部署,docker-entrypoint-initdb.d,数据迁移,自动化部署)

一、概述 MongoDB是一种常见的Nosql数据库&#xff08;非关系型数据库&#xff09;&#xff0c;以文档&#xff08;Document&#xff09;的形式存储数据。是非关系型数据库中最像关系型数据库的一种。本篇主要介绍下部署和数据迁移。 在 MongoDB 官方镜像部署介绍中&#xff…

MES系统通过eDrawings Pro API开发图纸批量转换工具,实现3D在线查看

声明&#xff1a;部分代码来源于网络&#xff0c;如有疑问&#xff0c;请联系本人删除。 通过C#结合eDrawings API提供接口&#xff0c;实现图纸转换为换.jpg、.tif、.bmp、.stl、.exe、.html、.zip、.edrw、.eprt 和 .eas格式工具&#xff0c;尤其是.html格式&#xff0c;可以…

Java阶段三06

第3章-第6节 一、知识点 理解MVC三层模型、理解什么是SpringMVC、理解SpringMVC的工作流程、了解springMVC和Struts2的区别、学会使用SpringMVC封装不同请求、接收参数 二、目标 理解MVC三层模型 理解什么是SpringMVC 理解SpringMVC的工作流程 学会使用SpringMVC封装请求…

【计算机网络】期末速成(2)

部分内容来源于网络&#xff0c;侵删~ 第五章 传输层 概述 传输层提供进程和进程之间的逻辑通信&#xff0c;靠**套接字Socket(主机IP地址&#xff0c;端口号)**找到应用进程。 传输层会对收到的报文进行差错检测。 比特流(物理层)-> 数据帧(数据链路层) -> 分组 / I…