深度剖析责任链模式

一、责任链模式的本质:灵活可扩展的流水线处理

责任链模式(Chain of Responsibility Pattern)是行为型设计模式的代表,其核心思想是将请求的发送者与接收者解耦,允许多个对象都有机会处理请求。这种模式完美解决了以下场景痛点:

  1. 动态流程编排:审批流程、风控流程等需要灵活调整顺序

  2. 请求分发机制:日志处理、权限校验等需要多层级过滤

  3. 未知处理者:异常捕获链、HTTP中间件等场景

UML类图


二、责任链模式的三种经典实现

1. 基础链表实现(传统方式)

// 处理器接口
public interface OrderHandler {void handle(Order order);void setNext(OrderHandler next);
}// 抽象基类
public abstract class AbstractOrderHandler implements OrderHandler {private OrderHandler next;@Overridepublic void setNext(OrderHandler next) {this.next = next;}protected void handleNext(Order order) {if (next != null) {next.handle(order);}}
}// 具体处理器
public class InventoryCheckHandler extends AbstractOrderHandler {@Overridepublic void handle(Order order) {if (!checkInventory(order)) {throw new RuntimeException("库存不足");}System.out.println("库存校验通过");handleNext(order);}private boolean checkInventory(Order order) {// 库存检查逻辑return true;}
}public class PaymentHandler extends AbstractOrderHandler {@Overridepublic void handle(Order order) {processPayment(order);System.out.println("支付处理完成");handleNext(order);}private void processPayment(Order order) {// 支付处理逻辑}
}

2. 集合迭代实现(Spring风格)

// 处理器接口
public interface Filter {void doFilter(Request request, Response response, FilterChain chain);
}// 链式调用容器
public class FilterChain {private List<Filter> filters = new ArrayList<>();private int index = 0;public FilterChain addFilter(Filter filter) {filters.add(filter);return this;}public void doFilter(Request request, Response response) {if (index < filters.size()) {Filter filter = filters.get(index++);filter.doFilter(request, response, this);}}
}// 使用示例
FilterChain chain = new FilterChain().addFilter(new AuthFilter()).addFilter(new LogFilter()).addFilter(new EncodingFilter());
chain.doFilter(request, response);

3. 注解驱动实现(企业级方案)

// 自定义注解
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface OrderProcessor {int order() default 0;String[] supportType();
}// 处理器基类
public abstract class BaseOrderProcessor {@Autowiredprivate ProcessorChain chain;public void process(OrderContext context) {if (canProcess(context)) {doProcess(context);}chain.process(context);}protected abstract boolean canProcess(OrderContext context);protected abstract void doProcess(OrderContext context);
}// 具体处理器
@OrderProcessor(order = 1, supportType = "NORMAL")
public class NormalOrderProcessor extends BaseOrderProcessor {@Overrideprotected boolean canProcess(OrderContext context) {return "NORMAL".equals(context.getOrderType());}@Overrideprotected void doProcess(OrderContext context) {// 普通订单处理逻辑}
}

三、责任链模式的五大应用场景

场景1:审批流程系统

  • 请假审批链(组长 → 经理 → 总监)

  • 采购审批链(金额分级审批)

  • 合同审批链(法务 → 财务 → CEO)

场景2:Web请求处理

  • Servlet Filter链

  • Spring Interceptor链

  • 网关过滤器链(限流 → 鉴权 → 日志)

场景3:订单处理系统

  • 订单校验链(库存 → 优惠券 → 地址)

  • 订单履约链(拆单 → 分配仓库 → 物流)

场景4:异常处理系统

  • 异常捕获链(业务异常 → 系统异常 → 全局异常)

  • 错误恢复链(重试 → 降级 → 告警)

场景5:游戏事件处理

  • 技能释放链(蓝量检查 → CD检查 → 效果触发)

  • 伤害计算链(防御计算 → 暴击计算 → 元素反应)


四、责任链模式与相关模式对比

模式关注点与责任链的区别
装饰器模式增强对象功能责任链强调传递,装饰器强调叠加功能
命令模式请求封装责任链处理请求传递,命令模式处理请求封装与执行
组合模式树形结构责任链是线性结构,组合模式是树状结构
策略模式算法替换责任链多个处理器协作,策略模式单个算法替换

五、企业级实战:Spring风格审批系统

架构设计

[审批请求] → [审批链构建器] → [审批处理器1] → [审批处理器2] → [...] → [审批完成]↑[审批规则配置中心]↑[数据库/配置中心]

完整代码实现

// 审批处理器接口
public interface ApprovalHandler {ApprovalResult handle(ApprovalContext context);void setNext(ApprovalHandler next);boolean support(ApprovalType type);
}// 链式处理器基类
public abstract class AbstractApprovalHandler implements ApprovalHandler {private ApprovalHandler next;private ApprovalType supportType;protected AbstractApprovalHandler(ApprovalType supportType) {this.supportType = supportType;}@Overridepublic void setNext(ApprovalHandler next) {this.next = next;}@Overridepublic ApprovalResult handle(ApprovalContext context) {if (!support(context.getType())) {return next.handle(context);}ApprovalResult result = doHandle(context);if (result.isApproved() && next != null) {return next.handle(context);}return result;}@Overridepublic boolean support(ApprovalType type) {return this.supportType == type;}protected abstract ApprovalResult doHandle(ApprovalContext context);
}// 具体审批处理器
public class DepartmentManagerHandler extends AbstractApprovalHandler {public DepartmentManagerHandler() {super(ApprovalType.DEPARTMENT_MANAGER);}@Overrideprotected ApprovalResult doHandle(ApprovalContext context) {// 部门经理审批逻辑return new ApprovalResult(true, "部门经理审批通过");}
}// 审批链构建工厂
@Service
public class ApprovalChainFactory {@Autowiredprivate List<ApprovalHandler> handlers;public ApprovalHandler createChain(ApprovalType type) {List<ApprovalHandler> sortedHandlers = handlers.stream().filter(h -> h.support(type)).sorted(Comparator.comparingInt(h -> h.getOrder())).collect(Collectors.toList());ApprovalHandler head = null;ApprovalHandler current = null;for (ApprovalHandler handler : sortedHandlers) {if (head == null) {head = handler;current = handler;} else {current.setNext(handler);current = handler;}}return head;}
}

六、责任链模式的五大优化技巧

1. 动态配置链顺序

// 基于配置文件的链定义
@Configuration
public class ChainConfig {@Beanpublic FilterChain securityFilterChain() {return new FilterChain().addFilter(new CorsFilter()).addFilter(new AuthFilter()).addFilter(new RateLimitFilter());}
}

2. 中断机制优化

public interface Handler {enum Result {CONTINUE, BREAK}Result handle(Request request);
}public class Chain {public void process(Request request) {for (Handler handler : handlers) {if (handler.handle(request) == Result.BREAK) {return;}}}
}

3. 性能监控埋点

public class MonitoredHandler implements Handler {private final Handler delegate;private final MeterRegistry registry;public MonitoredHandler(Handler delegate, MeterRegistry registry) {this.delegate = delegate;this.registry = registry;}@Overridepublic Result handle(Request request) {Timer.Sample sample = Timer.start(registry);try {return delegate.handle(request);} finally {sample.stop(registry.timer("handler.time", "handler", delegate.getClass().getSimpleName()));}}
}

七、常见陷阱与解决方案

陷阱现象解决方案
循环引用链式调用死循环增加最大调用深度检测
处理器遗漏请求未被任何处理器处理设置默认处理器或抛出明确异常
顺序依赖错误处理器执行顺序不符合预期使用优先级注解明确顺序
性能瓶颈长链路导致延迟过高引入异步处理或并行执行
状态污染处理器之间共享可变状态使用ThreadLocal或深拷贝上下文

 

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

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

相关文章

服务器使用centos7.9操作系统前需要做的准备工作

文章目录 前言1.操作记录 总结 前言 记录一下centos7.9操作系统的服务器在部署业务服务之前需要做的准备工作。 大家可以复制到自己的编辑器里面&#xff0c;有需求的注释一些步骤。 备注&#xff1a;有条件的项目推荐使用有长期支持的操作系统版本。 1.操作记录 # 更换阿里云…

Aitken 逐次线性插值

Aitken 逐次线性插值 用 Lagrange 插值多项式 L n ( x ) L_n(x) Ln​(x)计算函数近似值时&#xff0c;如需增加插值节点&#xff0c;那么原来算出的数据均不能利用&#xff0c;必须重新计算。为克服这个缺点&#xff0c;可用逐次线性插值方法求得高次插值。 令 I i 1 , i 2…

HARCT 2025 分论坛9:专用设备和机器人系统

会议名称&#xff1a;机电液一体化与先进机器人控制技术国际会议 会议简称&#xff1a;HARCT 2025 大会时间&#xff1a;2025年3月28日-30日 大会地点&#xff1a;中国桂林 主办单位&#xff1a;桂林航天工业学院、广西大学、桂林电子科技大学、桂林理工大学 协办单位&…

建筑兔零基础自学python记录18|实战人脸识别项目——视频检测07

本次要学视频检测&#xff0c;我们先回顾一下图片的人脸检测建筑兔零基础自学python记录16|实战人脸识别项目——人脸检测05-CSDN博客 我们先把上文中代码复制出来&#xff0c;保留红框的部分。 ​ 然后我们来看一下源代码&#xff1a; import cv2 as cvdef face_detect_demo(…

图书管理项目(spring boot + Vue)

想要该项目的话&#xff0c;就 jia 我&#xff0c;并在评论区给我说一下&#xff0c;只需要1元&#xff0c;我把整个项目发给你 jia微&#xff1a;18439421203&#xff08;名字叫&#xff1a;Bingo&#xff09; 运行图片&#xff1a;

Kubernetes 最佳实践:Top 10 常见 DevOps/SRE 面试问题及答案

1. 如何在 Kubernetes 中设置资源请求和限制&#xff1f; 资源请求确保容器有最小资源量&#xff08;CPU/内存&#xff09;&#xff0c;而限制则强制容器消耗的最大资源量。这有助于高效资源分配并防止资源争用。 示例&#xff1a; resources:requests:memory: "256Mi&…

java练习(19)

ps:练习来自力扣 给你一个整数数组 nums &#xff0c;其中元素已经按 升序 排列&#xff0c;请你将其转换为一棵 平衡 二叉搜索树。 // 定义二叉树节点类 class TreeNode {int val;TreeNode left;TreeNode right;TreeNode() {}TreeNode(int val) { this.val val; }TreeNode…

计算机考研复试上机05

目录 9、数学问题 1)进制转换 1.二进制数(北京邮电大学复试上机题) 2.进制转换(清华大学复试上机题) 3.十进制与二进制(清华大学复试上机题) 4.进制转换 2(清华大学复试上机题) 5.八进制(华中科技大学复试上机题) 6.又一版 A + B(浙江大学复试上机题) 7.…

LTSPICE仿真电路:(二十三)单端信号转差分信号的简单仿真

1.单端转差分 简单的可以用运放输出再加一个反向比例放大器即可&#xff0c;但是今天仿真一个其他的&#xff0c;在你好放大器上看到的电路。 2.电路图 仿真结果 交叉式单端转差分 优点&#xff1a;可以很明显看出来只需要用单电源就可以了&#xff0c;但是电容是不可缺少的…

Kotlin 2.1.0 入门教程(十七)接口

接口 接口可以包含抽象方法的声明&#xff0c;也可以包含方法的实现。 接口与抽象类的不同之处在于&#xff0c;接口无法存储状态。接口可以拥有属性&#xff0c;但这些属性要么必须是抽象的&#xff0c;要么就得提供访问器的实现。 接口使用 interface 关键字来定义&#x…

Python爬虫实战:获取51job职位信息,并做数据分析

注意&#xff1a;以下内容仅供技术研究&#xff0c;请遵守目标网站的robots.txt规定&#xff0c;控制请求频率避免对目标服务器造成过大压力&#xff01; 1. 环境准备 python import requests from bs4 import BeautifulSoup import pandas as pd import re import matplotl…

DeepSeek 又复活了!第三方平台接入DeepSeek R1

目录 1、秘塔搜索 2、硅基流动 3、腾讯元宝&#xff1a;强烈推荐 5 纳米AI 4、其它平台 评论区留言获取&#xff1a;DeepSeek-R1论文中文翻译文档PDF 往期精彩 1、秘塔搜索 https://metaso.cn/ 优点&#xff1a;满血版R1&#xff0c;操作方便缺点&#xff1a;无法关闭…

AN 433:源同步接口的约束与分析

文章目录 简介时钟和数据的关系SDR&#xff08;单数据速率&#xff09;和 DDR&#xff08;双数据速率&#xff09;接口约束默认时序分析行为 源同步输出输出时钟输出时钟约束时钟电路和约束示例 以系统为中心的输出延迟约束输出最大延时输出最小延时 以系统为中心的输出时序例外…

IO流-节点流与处理流

节点流和处理流的区别和联系 1. 节点流使底层流/低级流&#xff0c;是直接与数据源相接 2.处理流(包装流)包装节点流&#xff0c;既可以消除不同节点流的实现差异&#xff0c;也可以提供更方便的处理方法完成输入输出 3.处理流(包装流)对节点流进行包装&#xff0c;使用了修…

国密算法SM1、SM2、SM3和SM4 具体的使用和区别

国密算法是中国自主研发的密码算法&#xff0c;包括SM1、SM2、SM3和SM4&#xff0c;分别用于不同场景。以下是它们的具体使用和区别&#xff1a; SM1 对称加密算法 类型: 对称加密 密钥长度: 128位 使用场景: 用于数据加密和解密&#xff0c;适用于金融、政务等领域。 特点: …

【Prometheus】prometheus结合cAdvisor监控docker容器运行状态,并且实现实时告警通知

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全…

vue2中 computed 计算属性

文章目录 vue2中 computed 计算属性1. 什么是计算属性&#xff1f;2. 基本用法1. 定义计算属性2. 计算属性的缓存特性 3. 计算属性的高级用法1. 计算属性的 Getter 和 Setter 方法2. 计算属性的依赖追踪 4. 计算属性与方法的区别5. 实际应用案例1. 格式化数据2. 计算总价3. 动态…

Elasticsearch:同义词在 RAG 中重要吗?

作者&#xff1a;来自 Elastic Jeffrey Rengifo 及 Toms Mura 探索 RAG 应用程序中 Elasticsearch 同义词的功能。 同义词允许我们使用具有相同含义的不同词语在文档中搜索&#xff0c;以确保用户无论使用什么确切的词语都能找到他们所寻找的内容。你可能会认为&#xff0c;由于…

【devops】 Git仓库如何fork一个私有仓库到自己的私有仓库 | git fork 私有仓库

一、场景说明 场景&#xff1a; 比如我们Codeup的私有仓库下载代码 放入我们的Github私有仓库 且保持2个仓库是可以实现fork的状态&#xff0c;即&#xff1a;Github会可以更新到Codeup的最新代码 二、解决方案 1、先从Codeup下载私有仓库代码 下载代码使用 git clone 命令…

LabVIEW与小众设备集成

在LabVIEW开发中&#xff0c;当面临控制如布鲁克OPUS红外光谱仪这类小众专业设备的需求&#xff0c;而厂家虽然提供了配套软件&#xff0c;但由于系统中还需要控制其他设备且不能使用厂商的软件时&#xff0c;必须依赖特定方法通过LabVIEW实现设备的控制。开发过程中&#xff0…