责任链模式
责任链模式
责任链模式(Chain of Responsibility Pattern)是一种行为型设计模式,它用于将请求的发送者和接收者解耦,使多个对象都有机会处理请求。这种模式建立在一个处理对象的链上,每个处理对象都可以选择处理请求或者将请求传递给链上的下一个处理对象。
1、责任链模式角色
在 Java 中,责任链模式的实现通常包括以下几个要素:
- 抽象处理者(Handler):定义一个处理请求的接口,通常包括一个处理方法(例如:handleRequest())和一个设置下一个处理者的方法(例如:setNextHandler())。抽象处理者可以是一个接口或者抽象类。
- 具体处理者(ConcreteHandler):实现抽象处理者接口,具体处理不同的请求。如果当前处理者无法处理请求,它可以将请求传递给下一个处理者。
- 责任链(Chain ofResponsibility):将一系列的处理者连接成一个链,形成处理请求的链式结构。通常在客户端代码中构建这个责任链,并将请求从链的开头传递给第一个处理者。
2、责任链模式适用业务场景
- 日志记录系统:在系统中实现日志记录功能时,可以使用责任链模式。不同的日志级别(如调试、信息、警告、错误)可以由不同的处理器来处理,从而根据需要将日志记录到不同的目标(文件、数据库、控制台等)。
- 审批流程:在企业应用中,审批流程可能涉及多个层级的审批,每个层级的审批者都可以决定是否通过审批。责任链模式可以用于构建这样的审批流程,其中每个处理器代表一个审批者,如果一个审批者无法处理,请求将传递给下一个审批者。
- 安全认证:在安全认证中,可以使用责任链模式来实现多层级的认证机制。每个认证处理器可以负责一个特定的认证方法(如用户名密码、指纹、令牌等)。如果一个认证处理器无法通过认证,系统可以继续尝试下一个处理器。
- HTTP 请求处理:在 Web 应用程序中,HTTP
请求处理可以分成多个环节,例如身份验证、授权、输入验证、缓存等。责任链模式可以用于将每个环节拆分成一个处理器,并将请求从一个处理器传递到下一个处理器。 - 异常处理链:在处理异常时,可能需要一系列处理步骤来处理不同类型的异常。责任链模式可以用于将异常处理逻辑分解成多个处理器,每个处理器负责处理一种类型的异常。
- 请求过滤器:在 Web
开发中,请求过滤器可以用于对请求进行预处理,例如请求参数验证、安全检查等。责任链模式可以用于将不同的过滤逻辑拆分成多个处理器,依次对请求进行处理。 - 游戏闯关系统:在游戏中,角色的闯关系统可以采用责任链模式。每层关卡可以由不同的处理器来处理,如果一个处理器处理完成当前关卡,结果打到过关条件,系统可以使用下一个处理器进行下一关处理。
总之,责任链模式在任何需要将处理逻辑拆分成独立步骤,并且这些步骤可以灵活组合的情况下都是有用的。它帮助减少耦合,使代码更加可扩展和可维护。
3、游戏闯关系统责任链模式的应用
需求描述
假设现在有一个闯关游戏,进入下一关的条件是上一关的分数要高于 xx:
游戏一共 3 个关卡
进入第二关需要第一关的游戏得分大于等于 90
进入第三关需要第二关的游戏得分大于等于 80
普通业务代码实现
//第一关
public class FirstPassHandler {public int handler(){System.out.println("第一关-->FirstPassHandler");return 80;}
}//第二关
public class SecondPassHandler {public int handler(){System.out.println("第二关-->SecondPassHandler");return 90;}
}//第三关
public class ThirdPassHandler {public int handler(){System.out.println("第三关-->ThirdPassHandler,这是最后一关啦");return 95;}
}//客户端
public class HandlerClient {public static void main(String[] args) {FirstPassHandler firstPassHandler = new FirstPassHandler();//第一关SecondPassHandler secondPassHandler = new SecondPassHandler();//第二关ThirdPassHandler thirdPassHandler = new ThirdPassHandler();//第三关int firstScore = firstPassHandler.handler();//第一关的分数大于等于90则进入第二关if(firstScore >= 90){int secondScore = secondPassHandler.handler();//第二关的分数大于等于80则进入第二关if(secondScore >= 80){thirdPassHandler.handler();}}}
}
那么如果这个游戏有 99 关,我们的代码很可能就会写成这个样子:
if(第1关通过){// 第2关 游戏if(第2关通过){// 第3关 游戏if(第3关通过){// 第4关 游戏if(第4关通过){// 第5关 游戏if(第5关通过){// 第6关 游戏if(第6关通过){//...}}} }}
}
这种代码不仅冗余,并且当我们要将某两关进行调整时会对代码非常大的改动,这种操作的风险是很高的,因此,该写法非常糟糕。
责任链改造代码
如何解决这个问题,我们可以通过链表将每一关连接起来,形成责任链的方式,第一关通过后是第二关,第二关通过后是第三关…
这样客户端就不需要进行多重 if 的判断了:
public abstract class Handler {/*** 下一关用当前抽象接口来接收*/protected Handler next;public void setNext(Handler next) {this.next = next;}public abstract int handler();
}
public class FirstPassHandler extends Handler{final int firstPassScore = 90;private int play(){return firstPassScore;}@Overridepublic int handler(){System.out.println("第一关-->FirstPassHandler");int score = play();if(score >= firstPassScore){//分数>=firstPassScore 并且存在下一关才进入下一关if(this.next != null){return this.next.handler();}}return score;}
}
package com.lf.java.designpattern.chain;public class SecondPassHandler extends Handler{final int SecondPassScore = 80;private int play(){return SecondPassScore;}public int handler(){System.out.println("第二关-->SecondPassHandler");int score = play();if(score >= SecondPassScore){//分数>=SecondPassScore 并且存在下一关才进入下一关if(this.next != null){return this.next.handler();}}return score;}
}
package com.lf.java.designpattern.chain;public class ThirdPassHandler extends Handler{final int SecondPassScore = 70;private int play(){return SecondPassScore;}public int handler(){System.out.println("第三关-->ThirdPassHandler");int score = play();if(score >= SecondPassScore){//分数>=SecondPassScore 并且存在下一关才进入下一关if(this.next != null){return this.next.handler();}}return score;}
}
package com.lf.java.designpattern.chain;public class HandlerClient {public static void main(String[] args) {FirstPassHandler firstPassHandler = new FirstPassHandler();//第一关SecondPassHandler secondPassHandler = new SecondPassHandler();//第二关ThirdPassHandler thirdPassHandler = new ThirdPassHandler();//第三关// 和上面没有更改的客户端代码相比,只有这里的set方法发生变化,其他都是一样的firstPassHandler.setNext(secondPassHandler);//第一关的下一关是第二关secondPassHandler.setNext(thirdPassHandler);//第二关的下一关是第三关//说明:因为第三关是最后一关,因此没有下一关//从第一个关卡开始firstPassHandler.handler();}
}
改造完成的代码请求会从链的开头传递到每个处理器,根据请求的内容,每个处理器都可以选择处理请求或者将请求传递给下一个处理器。这样的设计使得责任链可以根据需要动态地调整和扩展。
但是还不能自动化的添加对应的链之间的关系。
责任链工厂改造代码
对于上面的请求链,我们也可以把这个关系维护到配置文件中或者一个枚举中。将使用枚举来动态的配置请求链并且将每个请求者形成一条调用链。
@Data
@AllArgsConstructor
@NoArgsConstructor
@EqualsAndHashCode
@ToString
public class PassEntity {/** 处理器顺序id */private Integer handlerId;/** 业务处理器名称*/private String name;/** 全限定类名 */private String conference;/** 前置处理器 */private Integer preHandlerId;/** 前置处理器 */private Integer nextHandlerId;
}
public enum PassEnum {// handlerId, 拦截者名称,全限定类名,preHandlerId,nextHandlerIdAPI_HANDLER(new PassEntity(1, "第一关", "com.lf.java.design.pattern.chain.FirstPassHandler", null, 2)),BLACKLIST_HANDLER(new PassEntity(2, "第二关", "com.lf.java.design.pattern.chain.SecondPassHandler", 1, 3)),SESSION_HANDLER(new PassEntity(3, "第三关", "com.lf.java.design.pattern.chain.ThirdPassHandler", 2, null)),;PassEntity passEntity;public PassEntity getPassEntity() {return passEntity;}PassEnum(PassEntity passEntity) {this.passEntity = passEntity;}
}
public interface IPassService {/*** 根据 handlerId 获取配置项* @param handlerId* @return*/PassEntity getPassEntity(Integer handlerId);/*** 获取第一个处理者* @return*/PassEntity getFirstPassEntity();
}
package com.lf.java.designpattern.chain;import java.util.HashMap;
import java.util.Map;public class PassServiceImpl implements IPassService {/*** 初始化,将枚举中配置的handler初始化到map中,方便获取*/private static Map<Integer, PassEntity> passEntityMap = new HashMap<>();static {PassEnum[] values = PassEnum.values();for (PassEnum value : values) {PassEntity passEntity = value.getPassEntity();passEntityMap.put(passEntity.getHandlerId(), passEntity);}}@Overridepublic PassEntity getPassEntity(Integer handlerId) {return passEntityMap.get(handlerId);}@Overridepublic PassEntity getFirstPassEntity() {for (Map.Entry<Integer, PassEntity> entry : passEntityMap.entrySet()) {PassEntity value = entry.getValue();// 没有上一个handler的就是第一个if (value.getPreHandlerId() == null) {return value;}}return null;}
}
package com.lf.java.designpattern.chain;public class PassHandlerEnumFactory {private static IPassService passService = new PassServiceImpl();// 提供静态方法,获取第一个handlerpublic static Handler getFirstPassHandler() {PassEntity firstPassEntity = passService.getFirstPassEntity();Handler firstPassHandler = newPassHandler(firstPassEntity);if (firstPassHandler == null) {return null;}PassEntity tempPassEntity = firstPassEntity;Integer nextHandlerId = null;Handler tempPassHandler = firstPassHandler;// 迭代遍历所有handler,以及将它们链接起来while ((nextHandlerId = tempPassEntity.getNextHandlerId()) != null) {PassEntity PassEntity = passService.getPassEntity(nextHandlerId);Handler PassHandler = newPassHandler(PassEntity);tempPassHandler.setNext(PassHandler);tempPassHandler = PassHandler;tempPassEntity = PassEntity;}// 返回第一个handlerreturn firstPassHandler;}/*** 反射实体化具体的处理者* @param firstPassEntity* @return*/private static Handler newPassHandler(PassEntity firstPassEntity) {// 获取全限定类名String className = firstPassEntity.getConference();try {// 根据全限定类名,加载并初始化该类,即会初始化该类的静态段Class<?> clazz = Class.forName(className);return (Handler) clazz.newInstance();} catch (ClassNotFoundException | IllegalAccessException | InstantiationException e) {e.printStackTrace();}return null;}}
测试业务类
public class PassHandlerClient {public static void main(String[] args) {Handler firstPassHandler = PassHandlerEnumFactory.getFirstPassHandler();firstPassHandler.handler();}
}
运行结果:
第一关-->FirstPassHandler
第二关-->SecondPassHandler
第三关-->ThirdPassHandler
这样通过动态的配置请求链就可以自动将每个请求者形成一条调用链。
还有更为复杂的链的形成,比如业务的链里面有复合链,而复合链又是普通的功能链组成的。这种责任链工厂方式更能充分体现出代码设计优势。
相关文章:

责任链模式
责任链模式 责任链模式(Chain of Responsibility Pattern)是一种行为型设计模式,它用于将请求的发送者和接收者解耦,使多个对象都有机会处理请求。这种模式建立在一个处理对象的链上,每个处理对象都可以选择处理请求或…...

【BI看板】Docker-compose安装Superset,安装最新版本2.1.0
软件及环境准备 docker, docker-compose docker-compose安装 字节码安装 #wget https://github.com/docker/compose/releases/download/v2.5.0/docker-compose-linux-x86_64 #mv docker-compose-linux-x86_64 docker-compose #chmod x /usr/local/bin/docker-com…...

VS2019生成的DLL,给QT(MinGW版本)使用的小结
VS2019端: a 基于生成一个DLL的工程(要注意生成是x86,还是x64的,需要和后面的QT的App工程对应),这里不多解释了,网上多的是; b 在cpp实现文件里,假如要导出一个这样的…...

c++--SLT六大组件之间的关系
1.SLT六大组件: 容器,迭代器,算法,仿函数,适配器,空间配置器 2.六大组件之间的关系 容器:容器是STL最基础的组件,没有容器,就没有数据,容器的作用就是用来存…...

解析个人信息保护影响评估
一、个人信息保护影响评估的概念及范围(What) 什么是“个人信息保护影响评估”?如何理解?“个人信息保护影响评估”的概念未在我国高位阶的法律规定中明确,其历经从观念到实践的演变,逐渐形成业界普遍认可…...
2.阿里云对象存储OSS
1.对象存储概述 文件上传,是指将本地图片、视频、音频等文件上传到服务器上,可以供其他用户浏览或下载的过程。文件上传在项目中应用非常广泛,我们经常发抖音、发朋友圈都用到了文件上传功能。 实现文件上传服务,需要有存储的支持…...

(三)Unity开发Vision Pro——入门
3.入门 1.入门 本节涵盖了几个重要主题,可帮助您加快visionOS 平台开发速度。在这里,您将找到构建第一个 Unity PolySpatial XR 应用程序的分步指南的链接,以及 PolySpatial XR 开发时的一些开发最佳实践。 2.开发与迭代 有关先决条件、开…...

召集令:CloudQuery 社区有奖征文活动来啦!
CloudQuery 社区第一期征文活动来袭!!!只要你对 CloudQuery 产品感兴趣,或者是希望了解 CQ ,都可以来参加,在本期活动中,我们也为大家准备了多种主题供你选择,CQ 使用案例、版本对比…...

【傅里叶级数与傅里叶变换】数学推导——1、基础知识点回顾及[Part1:三角函数的正交性]介绍
文章内容来自DR_CAN关于傅里叶变换的视频,本篇文章提供了一些基础知识点,比如三角函数常用的导数、三角函数换算公式等。 文章全部链接: 基础知识点 Part1:三角函数系的正交性 Part2:T2π的周期函数的傅里叶级数展开 P…...

BUUCTF [MRCTF2020]Ezpop解题思路
题目代码 Welcome to index.php <?php //flag is in flag.php //WTF IS THIS? //Learn From https://ctf.ieki.xyz/library/php.html#%E5%8F%8D%E5%BA%8F%E5%88%97%E5%8C%96%E9%AD%94%E6%9C%AF%E6%96%B9%E6%B3%95 //And Crack It! class Modifier {protected $var;publi…...

【IMX6ULL驱动开发学习】07.驱动程序分离的思想之平台总线设备驱动模型和设备树
一、驱动程序分离的思想 【IMX6ULL驱动开发学习】05.字符设备驱动开发模板(包括读写函数、poll机制、异步通知、定时器、中断、自动创建设备节点和环形缓冲区)_阿龙还在写代码的博客-CSDN博客 之前编写驱动程序的代码存在不少弊端:移植性差…...

深度学习中的python语法笔记总结
解释 torch中的 .clamp(min0) 在PyTorch中,torch.clamp将张量中的元素限制在指定的范围内。 torch.clamp(min0)会将张量中的每个元素与0进行比较,并将小于0的元素替换为0。其他大于等于0的元素则保持不变。 clamp函数原理 def clamp(x, lower, upper)…...

Reids 的整合使用
大家好 , 我是苏麟 , 今天带来强大的Redis . REmote DIctionary Server(Redis) 是一个由 Salvatore Sanfilippo 写的 key-value 存储系统,是跨平台的非关系型数据库。 Redis 是一个开源的使用 ANSI C 语言编写、遵守 BSD 协议、支持网络、可基于内存、分布式、可选…...

Vue3 —— watchEffect 高级侦听器
该文章是在学习 小满vue3 课程的随堂记录示例均采用 <script setup>,且包含 typescript 的基础用法 前言 Vue3 中新增了一种特殊的监听器 watchEffect,它的类型是: function watchEffect(effect: (onCleanup: OnCleanup) > void,o…...

Java异步子线程读取主线程参数的若干好玩场景
在开发过程中,我们难免会因为性能、实时响应等,需要异步处理的一些事务,并且在子线程中有时我们还需要获取主线程相关的参数。下面有若干方案可以实现上述场景,但会出现一定的问题。 场景1-基础场景 在主线程中开启子线程&#x…...

Android 视频开发
在 Android 平台上进行视频开发,您需要掌握以下关键知识点,以确保能够成功地开发和调试视频应用程序: Android视频架构: 了解 Android 的视频系统架构,包括视频捕获、编码、解码、渲染和显示等。 视频格式和编解码&am…...

【计算机网络篇】UDP协议
✅作者简介:大家好,我是小杨 📃个人主页:「小杨」的csdn博客 🐳希望大家多多支持🥰一起进步呀! UDP协议 1,UDP 简介 UDP(User Datagram Protocol)是一种无连…...

LeetCode 2682. 找出转圈游戏输家
【LetMeFly】2682.找出转圈游戏输家 力扣题目链接:https://leetcode.cn/problems/find-the-losers-of-the-circular-game/ n 个朋友在玩游戏。这些朋友坐成一个圈,按 顺时针方向 从 1 到 n 编号。从第 i 个朋友的位置开始顺时针移动 1 步会到达第 (i …...

数据结构单链表
单链表 1 链表的概念及结构 概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链 接次序实现的 。 在我们开始讲链表之前,我们是写了顺序表,顺序表就是类似一个数组的东西࿰…...

自定义WEB框架结合Jenkins实现全自动测试
自定义WEB框架结合Jenkins实现全自动测试 allure生成 allure生成 1.allure–纯命令运行 -固定的–稍微记住对应的单词即可。2 安装,2个步骤: 1.下载allure包,然后配置环境变量。 https://github.com/allure-framework/allure2/releases/tag/2.22.4 2.在…...

PHP加密与安全的最佳实践
PHP加密与安全的最佳实践 概述 在当今信息时代,数据安全是非常重要的。对于开发人员而言,掌握加密和安全的最佳实践是必不可少的。PHP作为一种常用的后端开发语言,提供了许多功能强大且易于使用的加密和安全性相关函数和类。本文将介绍一些P…...

SQL Server数据库无法连接
问题如下: 原因:sql server服务器未开启 解决方法:以管理员身份打开cmd,输入:net start mssqlserver。...

videojs 播放视频
背景:在项目中使用第三方插件videojs进行播放视频,点击事件更改播放的数据源。 一、视频相关理论 (一)、背景 网络流媒体的呈现形式分为两种:直播点播 (二)、流媒体的3种协议 分类:HTTPHLSRTMP定义:基于HTTP的流媒体…...

vue强制刷新变量
在前端开发中,我们经常需要变量的值实时响应到界面上。Vue就是一个非常强大的前端框架,它的数据绑定能够非常好地实现变量与界面的同步更新。但是有时候,我们需要强制刷新某个变量的值,以便界面能及时地反映出它的变化。本文将介绍…...

[QCA6174]QCA6174 5G WiFi DFS处理逻辑分析及雷达误检率高优化规避
DFS认证信息 WIFI DFS测试要求 Master设备需要测试的项目 4.6.2.1 Channel Availability Check 信道可用性检查 定义其作为雷达脉冲检测机制,当雷达脉冲出现时所占用的信道需要能被设备检测到已经被占用。当相关信道未被占用时,这些信道被称为Avaliable Channel可用信道 …...

预防SQL漏洞注入和规避网络攻击
前言: 虽然近些年SQL注入已经被各类的安全开发框架规避了绝大部分,但SQL注入作为一种最原始的攻击手段之一,破坏力仍然十分强大,因为它直捣黄龙数据中心。所以未雨绸缪,各位不可不重视。 预防SQL注入方法措施: 服务器…...

《Go 语言第一课》课程学习笔记(一)
配好环境:选择一种最适合你的 Go 安装方法 选择 Go 版本 一般情况下,建议采用最新版本。因为 Go 团队发布的 Go 语言稳定版本的平均质量一直是很高的,少有影响使用的重大 bug。可以根据不同实际项目需要或开源社区的情况使用不同的版本。 有…...

网络安全 Day29-运维安全项目-iptables防火墙
iptables防火墙 1. 防火墙概述2. 防火墙2.1 防火墙种类及使用说明2.2 必须熟悉的名词2.3 iptables 执行过程※※※※※2.4 表与链※※※※※2.4.1 简介2.4.2 每个表说明2.4.2.1 filter表 :star::star::star::star::star:2.4.2.2 nat表 2.5 环境准备及命令2.6 案例01:…...

SQL 复习 03
函数与关键字 用法说明round(x, n)四舍五入,x为浮点数,n为保留的位数ceil(x)向上取整floor(x)向下取整truncate(x, n)截断x,n为保留的位,该位之后的数值置零,位数表示示例:321.123,其中小数点前…...

出现 sudo: docker: command not found 的解决方法
目录 1. 问题所示2. 原理分析3. 解决方法3.1 未成功安装引起3.2 环境变量引起1. 问题所示 安装了docker,但是执行docker命令的时候,提示该问题: ubuntu@10-41-104-1:~$ sudo docker ps -a sudo: docker: command not foundubuntu@10-41-104-1:~$ sudo apt-get install doc…...