Java设计模式(五)—— 责任链模式
责任链模式定义如下:使多个对象都有机会处理请求,从而避免请求的发送者与接收者之间的耦合关系。将这些对象连成一条链,并沿着这条链传递该请求,知道有一个对象处理它为止。
适合使用责任链模式的情景如下:
- 有许多对象可以处理用户的请求,应用程序可自动确定谁处理用户请求
- 希望在用户不必明确指定接受者的情况下,向多个接受者提交一个请求
- 程序希望动态定制可处理用户请求的对象集合
一、问题的提出
在生活中我们经常会遇到这样的问题,例如:在企业工作的员工请假问题。假设假期少于一天的可由组长决定;多余1天少于2天的,可由车间主任决定;大于2天的可由经理决定。 “组长——主任——经理”构成了一个功能链。员工提出请假请求后,组长根据权限决定是否同意员工请假。若超出其权限,则传递给下一个责任人——主任。主任根据权限决定是否同意请假,若超出权限,在传递给下一个责任人——经理。
也就是说,请假这个“请求”,一定可以在功能链中的某一节点处理并返回。
二、责任链设计模式
由于是一系列类试图处理一个请求request,这些类之间是一个松散的耦合,唯一的共同点就是在他们之间传递请求request。
也就是说,来了一个请求,A类先处理,如果没有处理,就传递到B类处理,如果没有处理,就传递给C类处理,这就像一个链条一样传递下去。
责任链模式涉及的角色如下:
- 抽象处理者角色(Handler):定义一个处理请求的接口或抽象类。可定义一个方法,以设定和返回对下一节点的引用。
- 具体处理者(ConcreteHandler):具体处理者接到请求后,可以选择将请求处理完毕,或者将请求传给下一节点。由于具体处理者持有对下一节点的引用,因此,如果需要,具体处理者可以访问下一节点。
- 客户(Client):负责形成具体处理者的节点功能链,并传递初始请求。
利用责任链模式编制员工请假审批功能类:
(1)请求类
public class Request {int day; //请假的天数Request(int day) {this.day = day;}
}
(2)抽象处理者类Handler
public abstract class Handler {private Handler next;public Handler getNext() {return next;}public void setNext(Handler next) {this.next = next;}//定义抽象请求方法,子类要重写public abstract boolean handle(Request request);
}
(3)三个具体处理者类
public class ZuZhang extends Handler{static int limit = 1;@Overridepublic boolean handle(Request request) {if (request.day <= limit) {System.out.println("组长同意请假!");return true;}return getNext().handle(request);}
}
public class ZhuRen extends Handler{static int limit = 2;@Overridepublic boolean handle(Request request) {if (request.day <= limit) {System.out.println("主任同意请假!");return true;}return getNext().handle(request);}
}
public class JingLi extends Handler{static int limit = 3;@Overridepublic boolean handle(Request request) {if (request.day <= limit) {System.out.println("经理同意请假!");return true;}return getNext().handle(request);}
}
(4)生成责任链前后顺序关系类
public class MyChain {private Handler one = new ZuZhang();private Handler two = new ZhuRen();private Handler three = new JingLi();//生成责任链public void createChain() {one.setNext(two);two.setNext(three);}public void handle(Request request) {one.handle(request);}
}
(5)测试类
public class Test {public static void main(String[] args) {Request request = new Request(3);MyChain myChain = new MyChain();myChain.createChain();myChain.handle(request);}
}
测试结果:
经理同意请假!
三、反射的作用
在上面的MyChain类可以看出:所形成的责任链是刚性的,若需求分析发生了变化,链中需增加或减少节点,我们必须重新修改MyChain类,以适应需求分析发展的需要。那么,能否不修改程序,而又能满足需求分析的变化呢?答案是可以的,那就是“配置文件+反射”技术。
(1)对上面的例子而言,编写myconfig.txt配置文件:
chain=shejimoshi.zerenlian.ZuZhang,shejimoshi.zerenlian.ZhuRen,shejimoshi.zerenlian.JingLi
放在了这个位置:

(2)创建MyChain2
注释中很详细了写了每行代码的含义
import java.io.FileInputStream;
import java.util.Properties;public class MyChain2 {//声明了一个私有成员变量 handle,用于存储责任链中的各个节点。private Handler handle[];public void createChain() {try {//这里打开了一个配置文件 myconfig.txt,并使用 Properties 类加载其中的配置信息。FileInputStream in = new FileInputStream("D:\\Java_Dev\\IDEA_Projects\\Personal_project\\programmer_code\\src\\main\\java\\shejimoshi\\zerenlian\\myconfig.txt");Properties properties = new Properties();properties.load(in);///获取了配置文件中 chain 属性的值,并按照逗号分隔成一个字符串数组 unit,表示责任链中的节点。// 然后,代码通过数组的长度创建了一个 Handler 类型的数组 handle,用于存储责任链中的各个节点。String chain = properties.getProperty("chain");String unit[] = chain.split(",");int chainLength = unit.length;handle = new Handler[chainLength];//通使用Java反射机制,根据 unit 数组中的类名动态加载各个责任链节点对象,并存储到 handle 数组中。for (int i = 0; i < chainLength; i++) {handle[i] = (Handler) Class.forName(unit[i]).newInstance();}//通过遍历 handle 数组,设置责任链节点之间的前后关系。for (int i = 0; i < chainLength - 1; i++) {handle[i].setNext(handle[i+1]);}in.close();} catch (Exception e) {e.printStackTrace();}}//这个方法是责任链模式的核心,用于处理请求。这里直接调用责任链中的第一个节点来处理请求。public void handle(Request request) {handle[0].handle(request);}
}
(3)再次测试
public class Test {public static void main(String[] args) {Request request = new Request(2);MyChain2 myChain = new MyChain2();myChain.createChain();myChain.handle(request);}
}
测试结果:
主任同意请假!
(4)新增责任链
增加一个董事长的处理者,经理不能处理的时候转交给董事长处理
public class DongShiZhang extends Handler{static int limit = 5;@Overridepublic boolean handle(Request request) {if (request.day <= limit) {System.out.println("董事长同意请假!");return true;}return getNext().handle(request);}
}
(5)修改配置文件
增加董事长
chain=shejimoshi.zerenlian.ZuZhang,shejimoshi.zerenlian.ZhuRen,shejimoshi.zerenlian.JingLi,shejimoshi.zerenlian.DongShiZhang
(5)再次测试
public class Test {public static void main(String[] args) {Request request = new Request(4);MyChain2 myChain = new MyChain2();myChain.createChain();myChain.handle(request);}
}
测试结果:
董事长同意请假!
总结:可以看到,当需求发生变化时,只需要增加责任链中的具体处理者类,然后在配置文件中添加该具体处理者,并不需要去修改MyChain2的内容,这就是反射的作用。
相关文章:
Java设计模式(五)—— 责任链模式
责任链模式定义如下:使多个对象都有机会处理请求,从而避免请求的发送者与接收者之间的耦合关系。将这些对象连成一条链,并沿着这条链传递该请求,知道有一个对象处理它为止。 适合使用责任链模式的情景如下: 有许多对…...
VMLogin:虚拟浏览器提供的那些亮眼的功能
像VMLogin这样的虚拟浏览器具有多种功能,如安全的浏览环境、可定制的设置、跨平台的兼容性、更快的浏览速度、广告拦截等等。 虚拟浏览器的不同功能可以为您做什么? 使用虚拟浏览器是浏览互联网和完成其他任务的安全方式,没有风险。您可以在…...
第一个错误的版本
题目 你是产品经理,目前正在带领一个团队开发新的产品。不幸的是,你的产品的最新版本没有通过质量检测。由于每个版本都是基于之前的版本开发的,所以错误的版本之后的所有版本都是错的。 假设你有 n 个版本 [1, 2, …, n],你想找出…...
2023爱分析·AIGC市场厂商评估报告:拓尔思
AIGC市场定义 市场定义: AIGC,指利用自然语言处理技术(NLP)、深度神经网络技术(DNN)等人工智能技术,基于与人类交互所确定的主题,由AI算法模型完全自主、自动生成内容,…...
MobTech|场景唤醒的实现
什么是场景唤醒? 场景唤醒是moblink的一项核心功能,可以实现从打开的Web页面,一键唤醒App,并恢复对应的场景。 场景是指用户在App内的某个特定页面或状态,比如商品详情页、活动页、个人主页等。每个场景都有一个唯一…...
不在路由器上做端口映射,如何访问局域网内网站
假设现在外网有一台ADSL直接拨号上网的电脑,所获得的是公网IP。然后它想访问局域网内的电脑上面的网站,那么就需要在路由器上做端口映射。在路由器上做端口映射的具体规则是:将所有发向自己端口的数据,都转发到内网的计算机。 访…...
ChatGPT 辅助科研写作
前言 总结一些在科研写作中 ChatGPT 的功能,以助力提升科研写作的效率。 文章目录前言一、ChatGPT 简介1. ChatGPT 普通版与 Plus 版的区别1)普通账号2)Plus账号二、New Bing 简介1. 快速通过申请三、辅助学术写作1. 改写论文表述2. 语言润色…...
MySQL最大建议行数 2000w,靠谱吗?
本文已经收录到Github仓库,该仓库包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构、校招社招分享等核心知识点,欢迎star~ Github地址 1 背景 作为在后端圈开车的多年…...
【Tomcat 学习】
Tomcat 学习 笔记记录一、Tomcat1. Tomcat目录2. Tomcat启动3. Tomcat部署项目4. 解决Tomcat启动乱码问题5. JavaWeb项目创建部署6. 打war包发布项目7. Tomcat配置文件8. Tomcat配置虚拟目录(不用在webapps目录下)9. Tomcat配置虚拟主机10. 修改web项目默认加载资源路径一、Tom…...
重装系统如何做到三步装机
小白三步版在给电脑重装系统的过程中,它会提供系统备份、还原和重装等多种功能。下面也将介绍小白三步版的主要功能,以及使用技巧和注意事项。 主要功能 系统备份和还原:小白三步版可以帮助用户备份系统和数据,以防止重要数据丢失…...
蓝桥杯单片机第十一届省赛客观题(深夜学习——单片机)
第一场 (1)模电——》多级放大电路 阻容耦合,只通交流,不通直流。 变压器耦合,只通交流,不通直流。 光电耦合,主要是起隔离作用,更多的用在非线性的应用电路中 (2&a…...
Pandas对Excel文件进行读取、增删、打开、保存等操作的代码实现
文章目录前言一、Pandas 的主要函数包括二、使用步骤1.简单示例2.保存Excel操作3.删除和添加数据4.添加新的表单总结前言 Pandas 是一种基于 NumPy 的开源数据分析工具,用于处理和分析大量数据。Pandas 模块提供了一组高效的工具,可以轻松地读取、处理和…...
js常见的9种报错记录一下
js常见报错语法错误(SyntaxError)类型错误(TypeError)引用错误(ReferenceError)范围错误(RangeError)运行时错误(RuntimeError)网络错误(NetworkError)内部错误(InternalError)URI错误(URIError)eval错误&a…...
ORACLE not available报错处理办法
用sqlplus的时候 连接用户总是出现ORACLE not available 解决办法: 第一步: 请输入用户名: sys as sysdba 输入口令: 已连接到空闲例程。 第二步: 先连接到管理员用户下将用例开启 SQL> startup; ORACLE 例程已经启动。 然后就会出现一下 Total S…...
【Pandas】Python中None、null和NaN
经常混淆。 空值一般表示数据未知、不适用或将在以后添加数据。缺失值指数据集中某个或某些属性的值是不完整的。 一般空值使用None表示,缺失值使用NaN表示。 注意: python中没有null,但是有和其意义相近的None。 目录 1、None 2. NaN …...
线性表的学习
线性表定义 n个类型相同数据元素的有限序列,记作:a0,a1,a2,a3,...ai-1,ai,ai1...an-1(这里的0,1,2,3,i-1,i,i1,n-1都是元素的序号) 特点 除第一个元素无直接前驱。最后一个元素无直接后续&am…...
51单片机学习笔记_13 ADC
ADC 使得调节开发板上的电位器时,数码管上能够显示 AD 模块 采集电位器的电压值且随之变化。 开发板上有三个应用:光敏电阻,热敏电阻,电位器。 一般 AD 转换有多个输入,提高使用效率。 ADC 通过地址锁存与译码判断采…...
类和对象的基本认识之内部类
什么是内部类?当一个事物的内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服 务,那么这个内部的完整结构最好使用内部类。在 Java 中,可以将一个类定义在另一个类或者一个方法…...
【操作系统】进程和线程是什么之间是如何通信的
文章目录1、进程1.1、什么是进程1.2、进程的状态1.3、进程的控制结构1.4、进程的控制1.5、进程的上下文切换1.6、进程上下文切换场景1.7、进程间通信2、线程2.1、什么是线程2.2、线程的上下文切换2.3、线程间通信3、线程与进程的联系1、进程 1.1、什么是进程 进程(process) 是…...
setup、ref、reactive、computed
setup 理解:Vue3.0 中一个新的配置项,值为一个函数 setup 是所有 Composition API(组合API)“表演的舞台” 组件中所用到的数据、方法等,均要配置在 setup 中 setup 函数的两种返回值: 若返回一个对象…...
XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...
利用ngx_stream_return_module构建简易 TCP/UDP 响应网关
一、模块概述 ngx_stream_return_module 提供了一个极简的指令: return <value>;在收到客户端连接后,立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量(如 $time_iso8601、$remote_addr 等)&a…...
基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...
使用分级同态加密防御梯度泄漏
抽象 联邦学习 (FL) 支持跨分布式客户端进行协作模型训练,而无需共享原始数据,这使其成为在互联和自动驾驶汽车 (CAV) 等领域保护隐私的机器学习的一种很有前途的方法。然而,最近的研究表明&…...
Golang dig框架与GraphQL的完美结合
将 Go 的 Dig 依赖注入框架与 GraphQL 结合使用,可以显著提升应用程序的可维护性、可测试性以及灵活性。 Dig 是一个强大的依赖注入容器,能够帮助开发者更好地管理复杂的依赖关系,而 GraphQL 则是一种用于 API 的查询语言,能够提…...
定时器任务——若依源码分析
分析util包下面的工具类schedule utils: ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类,封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz,先构建任务的 JobD…...
【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...
C# SqlSugar:依赖注入与仓储模式实践
C# SqlSugar:依赖注入与仓储模式实践 在 C# 的应用开发中,数据库操作是必不可少的环节。为了让数据访问层更加简洁、高效且易于维护,许多开发者会选择成熟的 ORM(对象关系映射)框架,SqlSugar 就是其中备受…...
NFT模式:数字资产确权与链游经济系统构建
NFT模式:数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新:构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议:基于LayerZero协议实现以太坊、Solana等公链资产互通,通过零知…...
k8s业务程序联调工具-KtConnect
概述 原理 工具作用是建立了一个从本地到集群的单向VPN,根据VPN原理,打通两个内网必然需要借助一个公共中继节点,ktconnect工具巧妙的利用k8s原生的portforward能力,简化了建立连接的过程,apiserver间接起到了中继节…...
