设计模式-职责链模式

文章目录

  • 职责链模式
    • 模式概述
    • 主要角色
    • 适用场景
    • 实现步骤
    • 优点
    • 注意事项
  • 定义
  • 职责链结构
  • 示例
  • 总结

职责链模式

职责链模式是一种行为设计模式,它可以将请求的发送者和请求的处理者解耦,并按照预定义的顺序处理请求。职责链模式常用于需要逐级审批或转交处理的场景。

模式概述

职责链模式通过建立一个处理请求的对象链(职责链),将发送者和处理者分离开来。请求沿着职责链依次传递,直到有一个处理者能够处理它为止。这样,请求的发送者不需要知道哪个具体对象会处理请求,而处理者也不需要知道请求的发送者是谁。

主要角色

  • 抽象处理者(Handler):定义了处理请求的接口,并可以设定继任者(successor)。
  • 具体处理者(ConcreteHandler):实现抽象处理者的接口,根据自身的能力判断是否可以处理请求,如果可以则进行处理,否则将请求转发给继任者。

适用场景

适用于以下情况:

  • 系统中存在多个对象可以处理同一请求,但具体处理者在运行时才能确定。
  • 发送者和接收者之间的耦合关系需要降低。
  • 需要动态指定处理链的顺序或配置处理链。

实现步骤

  1. 定义抽象处理者,声明处理请求的方法,并提供设置继任者的方法。
  2. 实现具体处理者,根据自身的能力判断是否能够处理请求,如果可以则进行处理,否则将请求转发给继任者。
  3. 在客户端中创建处理链,并按照预定顺序设置继任者关系。
  4. 将请求发送到处理链的起始点,让处理链逐级处理请求。

优点

  • 解耦了请求的发送者和处理者,增强了系统的灵活性和可维护性。
  • 动态配置处理链,可以灵活地改变请求的处理顺序或增减处理者。
  • 符合单一职责原则和开闭原则。

注意事项

  • 需要正确设置处理链的顺序,确保请求能够被正确处理。
  • 处理链的最后一个处理者必须能够处理所有未被前面的处理者处理的请求,避免请求无法得到处理。

职责链模式可以简化请求的发送和处理过程,提高系统的灵活性和可扩展性。但需要合理设计和配置处理链,以确保请求能够被正确处理。

参考资料:《设计模式:可复用面向对象软件的基础》(GoF)

定义

职责链模式是一种行为设计模式,用于将请求的发送者和请求的处理者解耦,并且能够按照预定义的顺序处理请求。通过职责链模式,处理者负责处理请求,而客户只需将请求发送到职责链上,无需关心请求的处理细节和传递过程。

职责链结构

职责链模式的主要结构包括抽象处理者(Handler)和具体处理者(ConcreteHandler)。抽象处理者定义了一个处理请求的接口,并且可以设定继任者(successor)。具体处理者实现了抽象处理者的接口,并根据自身的能力来判断是否能够处理请求,如果可以处理则进行处理,否则将请求传递给继任者。
在这里插入图片描述

示例

以下是一个简单的职责链模式示例的代码:

// 设置处理请求的抽象类
abstract class Handler
{protected Handler successor;// 设置继任者public void SetSuccessor(Handler successor){this.successor = successor;}// 处理请求的抽象方法public abstract void HandleRequest(int request);
}// 具体处理者类
class ConcreteHandler1 : Handler
{public override void HandleRequest(int request){if (request >= 0 && request < 10){Console.WriteLine("{0} 处理请求 {1}", this.GetType().Name, request);}else if (successor != null){// 请求传递给继任者successor.HandleRequest(request);}}
}// 其他具体处理者类与第一个类似,范围不同// 客户端代码
static void Main(string[] args)
{Handler h1 = new ConcreteHandler1();Handler h2 = new ConcreteHandler2();Handler h3 = new ConcreteHandler3();h1.SetSuccessor(h2); // 设置继任者h2.SetSuccessor(h3);int[] requests = { 2, 4, 5, 7, 8, 12, 23, 27, 6 };foreach (int request in requests){h1.HandleRequest(request);}Console.Read();
}

在这里插入图片描述这段代码是一个简单的职责链模式的实现示例。以下是对这个代码的一些要点解释和改进建议:

  1. 职责链的顺序设置:在当前的代码中,通过SetSuccessor()方法手动设置了处理链的顺序。这种硬编码方式不够灵活,如果需要动态地配置处理链的顺序,可能需要修改代码。可以考虑使用其他方式,比如配置文件或依赖注入,从而实现更灵活的处理链配置。

  2. 请求的范围判断:在具体处理者类中,使用了一个条件判断语句来确定是否能够处理请求。目前的实现中,每个具体处理者都需要实现相同的条件判断逻辑,这样会导致代码重复。可以考虑将请求的范围判断逻辑抽离出来,单独封装成一个工具类或方法,让具体处理者直接调用,避免代码重复。

  3. 继任者的处理:当前的代码中,当具体处理者无法处理请求时,会将请求传递给继任者。这种方式可以确保请求能够被处理,但可能存在一个问题:如果继任者也无法处理请求,请求会一直传递下去直到最后一个处理者。为了避免请求无限传递,可以考虑在最后一个处理者中添加一个默认的处理逻辑,比如抛出一个异常或给出一个默认的处理结果。

  4. 扩展性和灵活性:职责链模式的一个重要特点是扩展性和灵活性。当前的代码通过继承和设置继任者的方式实现了处理链,这样可以方便地添加新的处理者并灵活调整处理顺序。但在实际应用中,可能还需要考虑更复杂的场景,比如动态添加、删除和调整处理者等。可以根据具体需求进行扩展和改进。

  5. 错误处理和异常处理:在当前的代码中,并没有专门处理处理请求时可能出现的错误或异常情况。为了提高代码的健壮性,可以在处理请求时添加适当的错误处理和异常处理机制,以确保系统的稳定性和可靠性。

在上述示例中,根据请求的大小范围,具体处理者对象判断是否可以处理该请求。如果可以处理,则进行处理;如果不能处理,则将请求传递给继任者进行处理。这样的处理过程可以按照设置好的顺序依次执行。

总结

职责链模式的优点包括解耦请求发送者和处理者、动态配置处理链、增强系统的灵活性和可扩展性等。但需要注意的是,如果没有正确配置处理链或最后一个处理者无法处理请求,请求可能无法得到处理。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

网络编程——套接字和字节序

目录 一、BSD套接字接口1.1 套接字类型1.2 套接字的位置 二、字节序2.1 大小端2.2 大小端判断2.3 主机字节序和网络字节序2.4 字节序转换函数 一、BSD套接字接口 BSD套接字接口是BSD的进程间通信的方式&#xff0c;它不仅支持各种形式的网络应用而且它还是一种进程间通信的机制…

汽车类 ±0.25°C SPI 温度传感器,TMP126EDBVRQ1、TMP126EDCKRQ1、TMP127EDBVRQ1引脚配置图

一、概述 TMP126-Q1 是一款精度为 0.25C 的数字温度传感器 &#xff0c; 支持的环境温度范围为 -55C 至 175C 。TMP126-Q1 具 有 14 位 &#xff08; 有符号 &#xff09; 温度分辨率(0.03125C/LSB)&#xff0c;并且可在 1.62V 至 5.5V 的电源电压范围内工作。TMP126-Q1 具有转…

Linux-crontab使用问题解决

添加定时进程 终端输入&#xff1a; crontab -e选择文本编辑方式&#xff0c;写入要运行的脚本&#xff0c;以及时间要求。 注意&#xff0c;如果有多个运行指令分两种情况&#xff1a; 1.多个运行指令之间没有耦合关系&#xff0c;分别独立&#xff0c;则可以直接分为两个…

SpringDataRedis 使用

1. SpringDataRedis 特点2. 使用 SpringDataRedis 步骤3. 自定义 RedisTemplate 序列化4. SpringDataRedis 操作对象 1. SpringDataRedis 特点 提供了对不同 Redis 客户端的整合&#xff08;Lettuce 和 Jedis&#xff09;提供了 RedisTemplate 统一 API 来操作 Redis支持 Redi…

【附安装包】Vm虚拟机安装Linux系统教程

软件下载 软件&#xff1a;Linux版本&#xff1a;18.0.4语言&#xff1a;简体中文大小&#xff1a;1.82G安装环境&#xff1a;VMware硬件要求&#xff1a;CPU2.0GHz 内存4G(或更高&#xff09;下载通道①丨百度网盘&#xff1a;1.Vm虚拟机15.5下载链接&#xff1a;https://pan…

Ansible学习笔记(一)

1.什么是Ansible 官方网站&#xff1a;https://docs.ansible.com/ansible/latest/installation_guide/intro_installation.html Ansible是一个配置管理和配置工具&#xff0c;类似于Chef&#xff0c;Puppet或Salt。这是一款很简单也很容易入门的部署工具&#xff0c;它使用SS…

【USRP】集成化仪器系列1 :信号源,基于labview实现

USRP 信号源 1、设备IP地址&#xff1a;默认为192.168.10.2&#xff0c;请勿 修改&#xff0c;运行阶段无法修改。 2、天线输出端口是TX1&#xff0c;请勿修改。 3、通道&#xff1a;0 对应RF A、1 对应 RF B&#xff0c;运行 阶段无法修改。 4、中心频率&#xff1a;当需要…

MySQL的共享锁和排他锁

锁定读 Locking Reads 有过编程语言并发学习经验的同学&#xff0c;应该都了解过读写锁的概念。读写锁主要是为了解决多读少写条件下&#xff0c;程序的并发性能问题。它的特点即是&#xff1a;如果一个线程持有了读锁&#xff0c;那么其他线程也是可以继续读取它锁定的数据&a…

CANOCO5.0实现冗余分析(RDA)最详细步骤

在地理及生态领域会常使用RDA分析&#xff0c;RDA的实现路径也有很多&#xff0c;今天介绍一下CANOCO软件的实现方法。 1.软件安装 时间调整到2010年 2.数据处理 得有不同的物种或者样点数值&#xff0c;再加上环境因子数据。 3.软件运行 4.结果解读 结果解读主要把握这几点…

Pytorch-以数字识别更好地入门深度学习

目录 一、数据介绍 二、下载数据 三、可视化数据 四、模型构建 五、模型训练 六、模型预测 一、数据介绍 MNIST数据集是深度学习入门的经典案例&#xff0c;因为它具有以下优点&#xff1a; 1. 数据量小&#xff0c;计算速度快。MNIST数据集包含60000个训练样本和1000…

【复杂网络建模】——ER网络和SF网络的阈值分析

目录 1、介绍ER网络和SF网络 2、计算网络阈值 2.1 ER&#xff08;Erdős-Rnyi&#xff09;网络 2.2 SF&#xff08;Scale-Free&#xff09;网络 3、 研究网络阈值的意义 1、介绍ER网络和SF网络 在复杂网络理论中&#xff0c;ER网络&#xff08;Erdős-Rnyi网络&#xff…

mybatis:动态sql【2】+转义符+缓存

目录 一、动态sql 1.set、if 2.foreach 二、转义符 三、缓存cache 1. 一级缓存 2. 二级缓存 一、动态sql 1.set、if 在update语句中使用set标签&#xff0c;动态更新set后的sql语句&#xff0c;&#xff0c;if作为判断条件。 <update id"updateStuent" pa…

【USRP】集成化仪器系列3 :频谱仪,基于labview实现

USRP 频谱仪 1、设备IP地址&#xff1a;默认为192.168.10.2&#xff0c;请勿 修改&#xff0c;运行阶段无法修改。 2、天线输出端口是TX1&#xff0c;请勿修改。 3、通道&#xff1a;0 对应RF A、1 对应 RF B&#xff0c;运行 阶段无法修改。 4、中心频率&#xff1a;当需要…

Unity 结构少继承多组合

为什么不推荐使用继承&#xff1f; 继承是面向对象的四大特性之一&#xff0c;用来表示类之间的 is-a 关系&#xff0c;可以解决代码复用的问题。虽然继承有诸多作用&#xff0c;但继承层次过深、过复杂&#xff0c;也会影响到代码的可维护性。所以&#xff0c;对于是否应该在…

程序员为什么要写bug,不能一次性写好吗?

仅仅听到“Bug”这个词就会让你作为一个开发人员感到畏缩。我们相信&#xff0c;优秀的程序员是那些编写无错误代码的人。随着一些开发人员强调要成为一名零错误程序员&#xff0c;我们进行了更深刻的思考&#xff0c;并发现事实的准确性。 所有制作的软件都应该没有错误。对此…

强化学习笔记

马尔科夫决策过程 markov chain&#xff1a; S \mathcal{S} S MRP&#xff1a; S &#xff0c; R \mathcal{S&#xff0c;R} S&#xff0c;R MDP&#xff1a; S &#xff0c; A ( s ) &#xff0c; R &#xff0c; P \mathcal{S&#xff0c;A(s)&#xff0c;R&#xff0c;P} …

在 Redis 中处理键值 | Navicat

Redis 是一个键值存储系统&#xff0c;允许我们将值与键相关联起来。与关系型数据库不同的是&#xff0c; 在Redis 中&#xff0c;不需要使用数据操作语言 &#xff08;DML&#xff09; 和查询语法&#xff0c;那么我们如何进行数据的写入、读取、更新和删除操作呢&#xff1f;…

怎么检测UI卡顿?(线上及线下)

什么是UI卡顿&#xff1f; 在Android系统中&#xff0c;我们知道UI线程负责我们所有视图的布局&#xff0c;渲染工作&#xff0c;UI在更新期间&#xff0c;如果UI线程的执行时间超过16ms&#xff0c;则会产生丢帧的现象&#xff0c;而大量的丢帧就会造成卡顿&#xff0c;影响用…

prometheus + grafana进行服务器资源监控

在性能测试中&#xff0c;服务器资源是值得关注一项内容&#xff0c;目前&#xff0c;市面上已经有很多的服务器资 源监控方法和各种不同的监控工具&#xff0c;方便在各个项目中使用。 但是&#xff0c;在性能测试中&#xff0c;究竟哪些指标值得被关注呢&#xff1f; 监控有…

springcloud-gateway简述

Spring Cloud Gateway 是一个用于构建 API 网关的项目&#xff0c;它是 Spring Cloud 生态系统中的一部分&#xff0c;旨在为微服务架构提供动态路由、负载均衡、安全性和监控等功能。 网关工程对应pom文件 <?xml version"1.0" encoding"UTF-8"?>…