责任链模式
责任链模式
责任链模式(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.在…...
Linux内存管理:malloc与free实现原理详解
Linux内存管理:malloc和free的实现原理深度解析1. 动态内存分配基础1.1 malloc和free函数原型void* malloc(size_t size); void free(void* ptr);malloc函数分配指定字节数的内存空间,返回指向该空间的void指针。由于返回的是通用指针,使用时…...
Rust 看了流泪,AI 看了沉默:扒开 Go 泛型最让你抓狂的“残疾”类型推断
大家好,我是Tony Bai。在这个大模型(AI)写代码如喝水一般简单的时代,你有没有遇到过一种极其憋屈的场景:你让 Claude Code 或者 Codex 帮你写了一段 Go 语言代码,逻辑清晰,结构优雅,…...
SkyWalking TTL配置实战:如何精准控制监控数据生命周期
1. 理解SkyWalking TTL的核心价值 当你的微服务集群每天产生TB级监控数据时,存储成本会像野马一样失控。去年我们一个电商项目就遇到过这样的困境——仅仅三个月ES集群就撑爆了200TB磁盘空间,而排查问题时发现99%的监控数据其实早已失效。这正是TTL&…...
如何构建可靠的HTML5解析测试框架:全面指南与最佳实践
如何构建可靠的HTML5解析测试框架:全面指南与最佳实践 【免费下载链接】gumbo-parser An HTML5 parsing library in pure C99 项目地址: https://gitcode.com/gh_mirrors/gum/gumbo-parser HTML5解析器是现代Web开发的核心组件,而构建一个可靠的测…...
Qwen3.5-4B-Claude-Opus惊艳效果展示:分步骤推导二分查找O(log n)全过程
Qwen3.5-4B-Claude-Opus惊艳效果展示:分步骤推导二分查找O(log n)全过程 1. 模型能力概览 Qwen3.5-4B-Claude-4.6-Opus-Reasoning-Distilled-GGUF是一个专为推理任务优化的轻量级模型,特别擅长处理需要分步骤分析的技术问题。这个4B参数的模型通过蒸馏…...
Python服务OOM频发真相:20年C Python内核开发者首曝智能体内存管理策略架构图(含perf+eBPF验证数据)
第一章:Python智能体内存管理策略架构总览Python智能体的内存管理并非简单复用CPython的引用计数与垃圾回收机制,而是构建在多层级抽象之上的协同式策略体系。该体系需同时满足短期对话上下文缓存、长期知识图谱嵌入存储、跨会话记忆检索以及隐私敏感数据…...
AtlasOS系统性能优化指南:从诊断到维护的全方位解决方案
AtlasOS系统性能优化指南:从诊断到维护的全方位解决方案 【免费下载链接】Atlas 🚀 An open and lightweight modification to Windows, designed to optimize performance, privacy and security. 项目地址: https://gitcode.com/GitHub_Trending/atl…...
Umi-OCR PDF文字识别全攻略:从技术原理到实战应用
Umi-OCR PDF文字识别全攻略:从技术原理到实战应用 【免费下载链接】Umi-OCR Umi-OCR: 这是一个免费、开源、可批量处理的离线OCR软件,适用于Windows系统,支持截图OCR、批量OCR、二维码识别等功能。 项目地址: https://gitcode.com/GitHub_T…...
COMSOL:模拟多尺度扩散模型下的流体运动与浆液扩散
COMSOL COMSOL模拟不同扩散模型下煤粒吸附/解吸扩 散 comsol仿真模拟电双层纳米电极,扩散双电层耦 合了Nernst-Planck方程和泊松方程。 comsol二 氧化碳混相驱替,多孔介质驱替,油气,扩散, 考虑浓度变化,速度…...
终极窗口置顶解决方案:如何用AlwaysOnTop告别窗口切换烦恼
终极窗口置顶解决方案:如何用AlwaysOnTop告别窗口切换烦恼 【免费下载链接】AlwaysOnTop Make a Windows application always run on top 项目地址: https://gitcode.com/gh_mirrors/al/AlwaysOnTop 你是否曾因频繁切换窗口而打断工作流?是否在多…...
