设计模式——责任链模式
文章目录
- 责任链模式的定义
- 场景示例
- 责任链模式实现方案
- 责任链模式扩展
- 责任链模式的优缺点
- 责任链模式在框架源码中的应用
责任链模式的定义
责任链模式又称职责链模式,是一种行为型设计模式。官方描述:使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。
原文链接或文中示例代码下载,请关注公众号【Qin的学习营地】,回复:【责任链模式】
场景示例
上述定义是对责任链模式这种思想的归纳总结,理解起来稍微抽象,下面结合一种场景逐步深入责任链模式。
企业采购
在企业中经常需要采购一些物品,小到一些廉价的生活用品,大到昂贵的精密器械,采购金额高低各异,不同采购金额需要由不同领导层次的管理人审批。比如金额在1万以下的只需部门负责人审批即可;金额超过1万且10万以下,部门负责人无权审批,需交由总经理审批;金额超过10万以上则只能层层上报到董事长,由董事长审批了,如下图。
简单的 if…else 实现方案
在上述企业采购中,采购金额不同分别交由不同层次的领导审批,这里最简单的实现方案就是使用 if…else 判断。代码如下:
// 采购审批处理类
public class PurchaseHandler {// 采购审批处理方法public String handleRequest(String purchaserName, Double purchaseAmount){if (purchaseAmount < 10000.0){ // 部门负责人审批return approvalByDepartmentLeader(purchaserName, purchaseAmount);} else if (purchaseAmount < 100000.0) { // 总经理审批return approvalByManager(purchaserName, purchaseAmount);} else { // 董事长审批return approvalByChairman(purchaserName, purchaseAmount);}}private String approvalByDepartmentLeader(String purchaserName, Double purchaseAmount){return (purchaserName + "申请采购金额:" + purchaseAmount + "元, 审批人:部门负责人");}private String approvalByManager(String purchaserName, Double purchaseAmount){return (purchaserName + "申请采购金额:" + purchaseAmount + "元, 审批人:总经理");}private String approvalByChairman(String purchaserName, Double purchaseAmount){return (purchaserName + "申请采购金额:" + purchaseAmount + "元, 审批人:董事长");}
}
if…else 实现方案存在的问题
分析 if…else 实现方案存在的问题:
- 代码结构不够清晰干净。
- 采购审批中的每个审批环节可能存在业务逻辑改动,比如部门负责人审核完之后,还是总经理签字,这就需要修改审批处理类和方法。
- 采购审批流程发生变化,当金额超过一定数额时,也不能完全由董事长审批,需要通过董事会投票决定,这就需要增加一个审批步骤,同样也需要修改源代码。
责任链模式实现方案
仔细分析上面的场景,客户端提出一个采购申请请求,对于客户端来说,他不需要关心是谁审批,后续审批流上可能是由部门负责人、总经理、董事长任一符合方进行审批。这样从部门负责人到总经理、再到董事长审批流程,可以看作是一条链,客户端提出的申请请求在这个链上传递,直到对应的责任人审批为止。
流程中可能在任意位置新增环节,每个环节可能会修改自己的处理逻辑,为了方便修改或复用每个环节;或者扩展新增环节,组合不同流程,可以将每个环节作为一个单独的对象,并且只实现单一的处理功能,这样细粒度划分功能,有利于降低修改的影响面和提高扩展性。
责任链模式概念
责任链模式就体现了上述细粒度抽取的思想,核心在于引入了一个抽象处理者。
责任链模式结构图:
责任链模式中的角色:
-
Handler(抽象处理者):一个处理请求的接口,一般设计为抽象类,其中定义一个处理请求的抽象方法,不同的具体处理者实现该方法;每个处理者可能需要将请求传递给下一个处理者,所以还可以定义一个处理者对象,作为后继者的引用,从而实现一条链。
-
ConcreteHandler(具体处理者):抽象处理者的具体实现子类,实现抽象处理类中的抽象方法,方法需要实现的两个功能:一是处理请求,处理具体处理者职责范围内的请求;二是转发请求,不在该具体处理者职责范围内的请求,就将请求转发给后继者处理。
-
Client(客户端):责任链客户端,组合处理者对象构造责任链,同时请求责任链处理请求对象。
组合的顺序很重要,如果顺序不对,处理的结果可能就不是符合要求的。
Client 不算责任链中的角色,外部使用责任链模式时自行创建。
责任链模式代码示例:
1、定义抽象处理者接口类,类中持有下一个处理者对象,同时实现请求处理方法:
// 定义责任链抽象接口类
public abstract class Handler {// 持有下一个处理者对象引用protected Handler nextHandler;// 设置下一个处理者对象public void setNextHandler(Handler nextHandler) {this.nextHandler = nextHandler;}// 处理采购申请请求public abstract String handleRequest(String purchaserName, Double purchaseAmount);
}
2、实现具体处理者类,实现处理请求的具体方法,方法内部逻辑:处理请求的具体逻辑,或者将请求对象传递给下一个处理者。这里实现三个具体处理者。
-
DepartmentLeaderHandler 部门负责人处理类
// 部门负责人处理类 public class DepartmentLeaderHandler extends Handler {@Overridepublic String handleRequest(String purchaserName, Double purchaseAmount) {// 职责范围内处理审批请求if (purchaseAmount < 10000){return (purchaserName + "申请采购金额:" + purchaseAmount + "元, 审批人:部门负责人");} else {// 职责范围外处理,交由下一个处理者处理if (this.nextHandler != null){return this.nextHandler.handleRequest(purchaserName, purchaseAmount);}}return null;} }
-
ManagerHandler 总经理处理类
// 总经理处理类 public class ManagerHandler extends Handler {@Overridepublic String handleRequest(String purchaserName, Double purchaseAmount) {// 职责范围内处理审批请求if (purchaseAmount < 100000){return (purchaserName + "申请采购金额:" + purchaseAmount + "元, 审批人:总经理");} else {// 职责范围外处理,交由下一个处理者处理if (this.nextHandler != null){return this.nextHandler.handleRequest(purchaserName, purchaseAmount);}}return null;} }
-
ChairmanHandler 董事长处理类
// 董事长处理类 public class ChairmanHandler extends Handler {@Overridepublic String handleRequest(String purchaserName, Double purchaseAmount) {// 职责范围内处理审批请求if (purchaseAmount >= 100000){return (purchaserName + "申请采购金额:" + purchaseAmount + "元, 审批人:董事长");} else {// 职责范围外处理,交由下一个处理者处理if (this.nextHandler != null){return this.nextHandler.handleRequest(purchaserName, purchaseAmount);}}return null;} }
3、客户端组合处理者构建责任链并调用
public class Client {public static void main(String[] args) {// 创建责任链上的每个处理者对象DepartmentLeaderHandler departmentLeaderHandler = new DepartmentLeaderHandler();ManagerHandler managerHandler = new ManagerHandler();ChairmanHandler chairmanHandler = new ChairmanHandler();// 组合处理者对象构造责任链departmentLeaderHandler.setNextHandler(managerHandler);managerHandler.setNextHandler(chairmanHandler);// 提交请求,调用责任链,调用责任链第一个处理者对象即可String result = departmentLeaderHandler.handleRequest("采购人员", 105000.0);System.out.println(result);}
}
当调用责任链处理请求时,并不知道谁会真正处理请求,只知道将请求提交到第一个处理者对象。从第一个处理者对象开始,整个责任链中的对象,要么自己处理请求,要么继续转发给下一个接收者。
从 if…else 判断到责任链模式的使用可以看出,代码结构变得清晰了。 责任链模式很好的满足了单一职责和开闭原则。
在责任链模式中,可能没有合适的处理者,请求不一定会被处理,所以可以在最后加上不支持处理该请求的提示。
责任链模式扩展
责任链构建方式
责任链构建方式多种多样,现在总结三种基本的实现方式:
1、客户端在提交请求前组合处理者构造责任链,即在使用的时候动态组合链。如前面责任链的实现就是这种。
2、可以在抽象处理者 Handler 内部实现责任链的组合。
3、由各个具体处理者自己决定后继者,这种方式要求每个处理者了解业务流程。
纯与不纯的责任链
纯的责任链模式
- 纯的责任链模式的每个处理者对象只会做两件事,要么处理请求并返回,要么将请求传递给链上的下一个处理者对象。
不纯的责任链模式
- 不满足纯的责任链模式要求的就是不纯的责任链,也可以称为功能链。一个请求在责任链中传递,每个处理者对象负责处理请求的某一方面的功能,处理完成后,不是停止,而是继续向后继者传递请求,后继者可以继续处理该请求。
- 在实际应用开发中,进行业务处理之前,通常需要进行权限检查、通用数据校验、数据逻辑校验等处理,然后才开始真正的业务逻辑实现。可以把这些功能分散到一个功能链中。这样做的目的是使程序结构更加灵活,而且复用性会更好。比如通用的权限检查只需要做一份,然后就可以在多个功能链中使用了。
责任链模式的优缺点
优点:
-
请求者和处理者松散耦合
在责任链模式中,请求者并不知道接收者是谁,也不知道具体如何处理,请求者只是负责向责任链发出请求就可以了。而每个处理者对象也不用管请求者或者是其他的处理者对象,只负责处理自己的部分,其他的就交给其他的处理者对象去处理。并且链中的对象不需要知道链的结构,可以由客户端负责链的创建,降低耦合。
-
动态组合
责任链模式会把功能处理分散到单独的处理者对象中,然后在使用的时候,可以动态组合处理者形构造责任链,从而可以灵活地给对象分配职责,也可以灵活地实现和改变对象的职责。
-
符合开闭原则
在系统中新增具体处理者对象时无需修改原代码,只需要在客户端重新构建责任链即可。
缺点:
-
产生很多细粒度对象
责任链模式会把功能处理分散到单独的处理者对象中,也就是每个处理者对象只处理一个方面的功能,要把整个业务处理完,需要很多处理者对象的组合,这样会产生大量的细粒度处理者对象。
-
不一定能被处理
责任链模式的每个处理者对象只负责自己处理的那一部分,因此可能会出现某个请求,把整个链传递完了,都没有处理者对象处理它。这就需要在使用责任链模式的时候,需要提供默认的处理,并且注意构建的链的有效性。
-
责任链模式可能会带来一些额外的性能损耗,因为它每次执行请求都要从链子开头开始遍历。甚至责任链构建不当,出现循环调用,导致系统陷入死循环。
责任链模式在框架源码中的应用
完整后续内容请关注公众号【Qin的学习营地】,回复:【责任链模式】
相关文章:

设计模式——责任链模式
文章目录 责任链模式的定义场景示例责任链模式实现方案责任链模式扩展责任链模式的优缺点责任链模式在框架源码中的应用 责任链模式的定义 责任链模式又称职责链模式,是一种行为型设计模式。官方描述:使多个对象都有机会处理请求,从而避免请…...
nginx得if语句内proxy_pass不允许携带url部分,如何处理
在nginx中,proxy_pass指令不能直接携带URL部分。但是,可以使用rewrite指令结合正则表达式来处理URL部分。 下面是一个示例配置,演示如何使用rewrite指令将URL中的某个部分进行替换后传递给后端服务器: location /v100/{proxy_…...
CentOS7设置 redis 开机自启动
CentOS7设置 redis 开机自启动 步骤1.创建redis.service文件2.重新加载所有服务3.设置开机自启动4.自由地使用linux系统命令4.1.启动 Redis 服务4.2.查看 Redis 状态(-l:查看完整的信息)4.3.停止 Redis 服务4.4.重启 Redis 服务 步骤 如果你傲娇,不想拷贝࿰…...

C++虚函数(定义,作用,原理,案例)
一.定义: C的虚函数是在父类(基类)中声明的的函数,它可在子类(派生类)中重写。二.作用 虚函数的目的是实现多态性,即在程序运行时根据对象的实际类型确定调用哪个函数。三.使用方法: 在基类中声明虚函数时,需要在函…...
C#中.NET 6.0 控制台应用通过EF访问新建数据库
目录 一、 操作步骤 二、编写EF模型和数据库上下文 三、 移植(Migrations)数据库 四、编写应用程序并运行 前文已经说过.NET Framework4.8 控制台应用通过EF访问新建数据库,这里的数据据库要根据事先编写好的EF模型、经过一番操作&#x…...
conda创建pytorch环境报错
昨天训练数据的时候,发现Anaconda占用C盘达到了20G(暑假在cmd状态下安装的,默认下载到了C盘),心道再创建几个环境,C盘就要爆红了,于是重装Anaconda到了D盘,不过之后的初始化并不顺利…...
数据结构-插入排序实现
文章目录 1、描述2、代码实现3、结果4、复杂度 1、描述 待排序的数组分为已排序、未排序两部分; 初始状态时,仅有第一个元素为已排序序列,第一个以外的元素为未排序序列; 此后遍历未排序序列, 将元素逐一插入到已排序的序列中&am…...
CGlib动态代理和JDK动态代理
CGlib代理模式是一种基于字节码操作的代理模式,它通过生成被代理类的子类来实现代理功能。 CGlib通过继承被代理类,生成一个代理类的子类,并重写父类的方法,在方法的前后插入相应的代理逻辑。这种方式不需要被代理类实现接口&…...

分类预测 | Matlab实现PSO-GRU-Attention粒子群算法优化门控循环单元融合注意力机制多特征分类预测
分类预测 | Matlab实现PSO-GRU-Attention粒子群算法优化门控循环单元融合注意力机制多特征分类预测 目录 分类预测 | Matlab实现PSO-GRU-Attention粒子群算法优化门控循环单元融合注意力机制多特征分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 1.Matlab实现PSO…...
Python OpenCV 视频抽帧处理并保存
上篇文章中基于OpenCV实现图像处理后,类似的,也可以对视频进行处理。OpenCV库可以将视频的每一帧读取出来,然后对每一帧图像做相应的操作,并保存成新的视频。 1. 读取视频,获取相关参数 import cv2 import numpy as…...

英伟达AI布局的新动向:H200 GPU开启生成式AI的新纪元
英伟达Nvidia是全球领先的AI计算平台和GPU制造商,近年来一直在不断推出创新的AI产品和解决方案,为各行各业的AI应用提供强大的支持。 最近,英伟达在GTC 2023大会上发布了一款专为训练和部署生成式AI模型的图形处理单元(GPU&#…...

Windows11 python3.12 安装pyqt6 pyqt6-tools
Windows11 python3.12 安装pyqt6比较容易,但pyqt6-tools一直安装不上去。出错信息如下: (venv) PS D:\python_project\pyqt6> pip install pyqt6-tools Collecting pyqt6-toolsUsing cached pyqt6_tools-6.4.2.3.3-py3-none-any.whl (29 kB) Collec…...

反弹Shell
概述 反弹shell(reverse shell)就是控制端监听在某TCP/UDP端口,被控端发起请求到该端口,并将其命令行的输入输出转到控制端。reverse shell与telnet,ssh等标准shell对应,本质上是网络概念的客户端与服务端…...
Guava RateLimiter的限流机制详解
限流是保护高并发系统的三种有效方法之一。另外两个分别是缓存和降级。限流在很多场景中都会使用到限制并发数和请求数。例如,在限时抢购的情况下,限流可以保护您自己的系统和下游系统不被巨大的流量淹没。 限流的目的是通过限制并发访问或请求或者限制…...
详解nginx的root与alias
在Nginx中,root和alias指令都可以用来指定Web服务器中的文件根目录。不过,它们之间有一些关键的区别。 root指令指定的是服务器根目录,是用于处理HTTP请求时所使用的默认根目录。例如,若root /var/www/html;,则访问htt…...
在HBuilderX中配置Vue Router的步骤
以下是在HBuilderX中配置Vue Router的步骤: 在项目中安装Vue Router,可以使用npm或yarn命令进行安装。 在src目录下创建routers.js文件,并在该文件中编写路由相关代码,例如: import Vue from vue import Router from …...
通过接口抓取公众号信息并群发
总体步骤 通过非官方接口,登陆公众号获取cookie、token通过token拼接需要的参数,请求被抓取的公众号列表数据通过列表数据获取文章内容解析文章内容并通过官方接口创建草稿通过非官方接口群发创建的草稿(非认证用户,已认证用户可以通过官方接…...

Python基础入门----如何通过conda搭建Python开发环境
文章目录 使用 conda 搭建Python开发环境是非常方便的,它可以帮助你管理Python版本、依赖库、虚拟环境等。以下是一个简单的步骤,演示如何通过 conda 搭建Python开发环境: 安装conda: 如果你还没有安装 conda,首先需要安装Anaconda或Miniconda。Anaconda是一个包含很多数据…...

计算机网络的体系结构
目录 一. 计算机体系结构的形成二. 协议与层次划分2.1 数据传输过程2.2 什么是网络协议2.3 网络协议的三要素2.4 协议有两种形式2.4 各层协议2.5 什么是复用和分用 \quad 一. 计算机体系结构的形成 \quad 计算机网络是一个非常复杂的系统, 相互通信的两个计算机系统必须高度协调…...

cesium雷达扫描(模糊圆效果)
cesium雷达扫描(模糊圆效果) 1、实现思路 使用ellipse方法加载圆型,修改ellipse中‘material’方法重写自己的glsl来实现当前效果 1、示例源码 index.html <!DOCTYPE html> <html lang="en"><head><!<...

XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...
进程地址空间(比特课总结)
一、进程地址空间 1. 环境变量 1 )⽤户级环境变量与系统级环境变量 全局属性:环境变量具有全局属性,会被⼦进程继承。例如当bash启动⼦进程时,环 境变量会⾃动传递给⼦进程。 本地变量限制:本地变量只在当前进程(ba…...
pam_env.so模块配置解析
在PAM(Pluggable Authentication Modules)配置中, /etc/pam.d/su 文件相关配置含义如下: 配置解析 auth required pam_env.so1. 字段分解 字段值说明模块类型auth认证类模块,负责验证用户身份&am…...

现代密码学 | 椭圆曲线密码学—附py代码
Elliptic Curve Cryptography 椭圆曲线密码学(ECC)是一种基于有限域上椭圆曲线数学特性的公钥加密技术。其核心原理涉及椭圆曲线的代数性质、离散对数问题以及有限域上的运算。 椭圆曲线密码学是多种数字签名算法的基础,例如椭圆曲线数字签…...
Axios请求超时重发机制
Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式: 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...

SpringTask-03.入门案例
一.入门案例 启动类: package com.sky;import lombok.extern.slf4j.Slf4j; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cache.annotation.EnableCach…...

初探Service服务发现机制
1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源…...

uniapp 开发ios, xcode 提交app store connect 和 testflight内测
uniapp 中配置 配置manifest 文档:manifest.json 应用配置 | uni-app官网 hbuilderx中本地打包 下载IOS最新SDK 开发环境 | uni小程序SDK hbulderx 版本号:4.66 对应的sdk版本 4.66 两者必须一致 本地打包的资源导入到SDK 导入资源 | uni小程序SDK …...
适应性Java用于现代 API:REST、GraphQL 和事件驱动
在快速发展的软件开发领域,REST、GraphQL 和事件驱动架构等新的 API 标准对于构建可扩展、高效的系统至关重要。Java 在现代 API 方面以其在企业应用中的稳定性而闻名,不断适应这些现代范式的需求。随着不断发展的生态系统,Java 在现代 API 方…...

水泥厂自动化升级利器:Devicenet转Modbus rtu协议转换网关
在水泥厂的生产流程中,工业自动化网关起着至关重要的作用,尤其是JH-DVN-RTU疆鸿智能Devicenet转Modbus rtu协议转换网关,为水泥厂实现高效生产与精准控制提供了有力支持。 水泥厂设备众多,其中不少设备采用Devicenet协议。Devicen…...