设计模式——Chain(责任链)设计模式

摘要

责任链设计模式是一种行为设计模式,通过链式调用将请求逐一传递给一系列处理器,直到某个处理器处理了请求或所有处理器都未能处理。它解耦了请求的发送者和接收者,允许动态地将请求处理职责分配给多个对象,支持请求的灵活传递或中断。

1. 责任链设计模式是什么

1.1. 责任链设计模式

责任链模式理解就是遍历某一个接口下所有的实现类中方法。责任链模式的核心思想就是通过链式调用将请求逐一传递给一系列处理器(实现了同一个接口的对象),直到某个处理器处理了请求,或者链中的所有处理器都未能处理该请求。

1.2. 责任链模式的基本工作原理:

  1. 处理者链:将多个处理者(通常是实现了相同接口的类)链接在一起,每个处理者负责处理请求的一个特定方面。
  2. 请求传递:请求会沿着责任链传递,每个处理者都会判断是否能够处理该请求,处理后返回或继续传递给下一个处理者。
  3. 终止条件:如果某个处理者能够处理该请求,链条上后续的处理者通常会被跳过,直接返回处理结果;如果没有处理者能处理,最终请求可能会失败。

1.3. 简化的责任链模式结构

可以把责任链模式理解为:

  • 链表结构:每个节点(处理器)持有对下一个节点的引用,依次处理请求。
  • 处理器接口:每个节点实现一个接口,负责具体的处理逻辑。

责任链模式(Chain of Responsibility Pattern)与 装饰器模式(Decorator Pattern)迭代器模式(Iterator Pattern) 在某些方面有相似之处,但它们的核心思想和使用场景略有不同。

特性

责任链模式

装饰器模式

目的

解耦请求的发送者和接收者,将请求交给多个处理者逐一处理

动态为对象添加功能,增强其行为

结构

请求按链式结构传递,每个处理者持有下一个处理者的引用

对原对象进行包装,增加额外功能

请求传递

请求沿着责任链传递,直到某个处理器处理请求或链终止

不传递请求,直接在原对象上增强功能

设计模式重点

请求处理的顺序和中断条件控制

通过包装原对象来动态增加功能

典型应用场景

多个处理器顺序处理请求,逐个判断并决定是否处理

动态为对象添加行为,功能扩展而不修改原对象代码

耦合性

请求发送者与接收者解耦,减少了依赖关系

可以灵活地为对象添加功能,扩展类的功能而不修改其代码

责任

每个处理者负责部分请求处理逻辑

每个装饰器负责对原对象的功能扩展

1.4. 责任链设计模式的作用

  1. 请求传递:每个处理者(链上的对象)都有一个引用指向下一个处理者。请求沿着链依次传递,直到被某个处理者处理或链结束。
  2. 解耦:发送方无需直接与处理方耦合,降低代码的复杂度。
  3. 可扩展性:可以灵活地添加、删除或修改链上的处理者,而不影响整体结构。
  4. 职责分离:每个处理者只关注自己能处理的部分,其余的请求传递给下一个处理者。

1.5. 责任链模式的优点

  1. 解耦请求与处理者:请求方不需要知道具体处理者是谁,降低耦合。
  2. 动态组合职责链:可以灵活地增加、删除或重排链中的处理节点。
  3. 代码清晰:将复杂的逻辑分解到多个处理节点中,职责单一。

1.6. 责任链模式的缺点

  1. 性能开销:当责任链过长时,可能带来性能问题。
  2. 调试困难:请求处理链条较长时,调试排查问题可能变得复杂。
  3. 不保证请求被处理:如果没有合适的处理者,可能导致请求最终未被处理。(执行链是有默认的顺序,可以借助Spring中@Order注解来实现有序)

2. 责任链模式类图实现

2.1. 责任链模式的结构

责任链模式主要包含以下角色:

  1. Handler(处理者抽象类/接口):
    • 定义处理请求的接口,并存储下一个处理者的引用。
  1. ConcreteHandler(具体处理者):
    • 继承或实现 Handler,处理它负责的请求。如果无法处理,则将请求转发给下一个处理者。
  1. Client(客户端):
    • 创建责任链,并向链头发出请求。

3. 责任链模式使用场景

3.1. 多级请求处理

  • 当一个请求需要经过多个对象处理,且不明确具体是由哪个对象处理时。
  • 示例:
    • 日志系统:根据日志级别(DEBUG、INFO、WARN、ERROR)动态决定由哪个日志处理器处理。
    • 审批流:如贷款审批流程,依次经过组长、经理、总监的审批。

3.2. 请求处理职责动态变化

  • 处理职责可能会随时增减,且希望能灵活调整处理顺序。
  • 示例:风控规则引擎:新增或调整规则处理节点时,动态维护责任链。

3.3. 避免请求发送者与接收者耦合

  • 当请求发送者不需要知道接收者的具体实现时,可以通过责任链解耦。
  • 示例:支付系统:如多种支付渠道(银行卡、第三方支付、余额支付)按优先级尝试支付。

3.4. 请求的处理逻辑可以被分解

  • 请求的处理逻辑非常复杂,且可以拆分成多个步骤时,每个步骤都可以成为责任链上的一个节点。
  • 示例:HTTP 请求拦截器:如在请求到达服务器前,依次经过身份认证、权限验证、数据校验等。

3.5. 需要支持请求的灵活传递或中断

  • 某个处理器完成后,可以继续传递给下一个处理器,也可以选择中止传递。
  • 示例:事件处理机制:如 Java 的事件监听器,事件沿着监听器链依次传递,直到某个监听器处理完成。

3.6. 责任链具体场景示例

3.6.1. 风控规则链

在信贷风控中,每个用户的申请需要经过一系列的规则检查,例如:

  • 身份证校验
  • 黑名单检测
  • 信用评分计算
  • 额度限制校验

这些规则可以按顺序依次处理,或者某个规则失败时中断处理。

3.6.2. API 请求过滤

如在微服务架构中,API 请求经过以下责任链:

  • 身份验证拦截器
  • 参数校验拦截器
  • 权限检查拦截器
  • 日志记录拦截器

3.6.3. 异常处理链

多个异常处理器按顺序处理异常,例如:

  • 数据库异常处理器
  • 网络异常处理器
  • 业务逻辑异常处理器
    责任链可以逐级定位和处理异常。

3.6.4. 广告投放系统

  • 按用户属性(年龄、性别、兴趣)动态匹配投放规则。
  • 若某规则不适用,则传递给下一个规则。

4. 责任链模式示例(Spring)

在 Spring 中使用 责任链模式 并让链上的处理器由 Spring 容器管理,可以通过以下方式实现:利用 Spring 的 @Component 注解 配合 自动注入(@Autowired) 或者通过 @Order 注解 实现责任链的顺序化管理。

4.1. 定义责任链接口

public interface Handler {boolean handle(Request request);
}

每个校验规则实现该接口,handle 方法返回 true 代表校验通过,false 代表校验失败。

4.2. 定义请求对象

@Data
public class Request {private String userId;private int creditScore;private double loanAmount;// 其他字段...// Getter & Setter
}

4.3. 创建具体的处理器

每个处理器(节点)实现 Handler 接口,例如:

4.3.1. 身份校验处理器

@Component
public class IdentityValidationHandler implements Handler {@Overridepublic boolean handle(Request request) {// 假设用户 ID 不能为空if (request.getUserId() == null || request.getUserId().isEmpty()) {System.out.println("身份校验失败");return false;}System.out.println("身份校验通过");return true;}
}

4.3.2. 黑名单检测处理器

@Component
public class BlacklistCheckHandler implements Handler {@Overridepublic boolean handle(Request request) {// 假设用户 "123" 是黑名单用户if ("123".equals(request.getUserId())) {System.out.println("黑名单校验失败");return false;}System.out.println("黑名单校验通过");return true;}
}

4.3.3. 信用评分校验处理器

public class CreditScoreCheckHandler implements Handler {@Overridepublic boolean handle(Request request) {// 假设信用评分不能低于 600if (request.getCreditScore() < 600) {System.out.println("信用评分校验失败");return false;}System.out.println("信用评分校验通过");return true;}
}

4.3.4. 额度检查处理器

@Component
public class LoanAmountCheckHandler implements Handler {@Overridepublic boolean handle(Request request) {// 假设贷款金额不能超过 50 万if (request.getLoanAmount() > 500000) {System.out.println("额度检查失败");return false;}System.out.println("额度检查通过");return true;}
}

4.4. 责任链测试

package com.zhuangxiaoyan.hyxftest.chain;import com.zhuangxiaoyan.hyxftest.HyxfTestApplication;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;/*** chaintest** @author xjl* @version 2024/12/04 19:56**/@SpringBootTest(classes = HyxfTestApplication.class)
public class ChainTest {@Autowiredprivate HandlerChain handlerChain;@Testpublic void chainTest() {handlerChain.process(new Request("122222", 500, 10000));}
}

5. 博文参考

《软件设计模式》

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

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

相关文章

macOS 15.1.1 (24B2091) 系统中快捷键符号及其代表的按键的对照表

以下是 macOS 15.1.1 (24B2091) 系统中快捷键符号及其代表的按键的对照表&#xff1a; 符号按键名称描述⌘Command (Cmd)常用的功能键&#xff0c;用于执行大多数快捷操作。⌥Option (Alt)Option 键&#xff0c;常用于辅助操作和特殊字符输入。⇧ShiftShift 键&#xff0c;常用…

el-table一键选择全部行,切换分页后无法勾选

el-table一键全选&#xff0c;分页的完美支持 问题背景尝试解决存在问题问题分析 解决方案改进思路如下具体代码实现如下 问题背景 现在有个需求&#xff0c;一个表格有若干条数据(假设数量大于20&#xff0c;每页10条&#xff0c;保证有2个以上分页即可)。 现在需要在表格上方…

【55 Pandas+Pyecharts | 实习僧网Python岗位招聘数据分析可视化】

文章目录 &#x1f3f3;️‍&#x1f308; 1. 导入模块&#x1f3f3;️‍&#x1f308; 2. Pandas数据处理2.1 读取数据2.2 查看数据信息2.3 去除重复数据2.4 调整部分城市名称 &#x1f3f3;️‍&#x1f308; 3. Pyecharts数据可视化3.1 招聘数量前20岗位3.2 各城市招聘数量3…

JAVA期末复习(30道填空题梳理知识点)

通过梳理一些常见的填空题&#xff0c;有效地复习知识点&#xff0c;帮助大家顺利通过考试。本文将总结30道典型的填空题&#xff0c;并分析其中涉及的知识点。 一、基本语法 JAVA程序的入口方法是&#xff1a; public static void main(String[] args) { }这个题目考察了JAVA…

C++11新特性之线程std::thread

C std::thread的定义和功能 std::thread是C11引入的标准库类&#xff0c;用于创建和管理线程。通过std::thread&#xff0c;程序可以并发执行多个任务&#xff0c;从而提高效率。 功能与作用&#xff1a; 创建线程&#xff1a;可以启动一个线程执行某个函数或任务。管理线程…

【赵渝强老师】PostgreSQL的控制文件

PostgreSQL数据库的物理存储结构主要是指硬盘上存储的文件&#xff0c;包括&#xff1a;数据文件、日志文件、参数文件、控制文件、WAL预写日志文件等等。 下面重点讨论一下PostgreSQL的控制文件。 视频讲解如下 【赵渝强老师】PostgreSQL的控制文件 控制文件记录了数据库运行…

在做题中学习(79):最小K个数

解法&#xff1a;快速选择算法 说明&#xff1a;堆排序也是经典解决问题的算法&#xff0c;但时间复杂度为&#xff1a;O(NlogK)&#xff0c;K为k个元素 而将要介绍的快速选择算法的时间复杂度为: O(N) 先看我的前两篇文章&#xff0c;分别学习&#xff1a;数组分三块&#…

【linux】shell(32)-循环控制

for循环 在 Shell 脚本中&#xff0c;for 循环是一种常见的循环结构&#xff0c;用于遍历列表、数组或命令输出。 基本语法 for 循环的基本语法如下&#xff1a; #!/bin/bash for variable in list docommands donevariable 是一个临时变量&#xff0c;用于存储每次迭代中的…

Pydantic 动态字段:使用和不使用 `@computed_field` 的对比指南

Pydantic 动态字段&#xff1a;使用和不使用 computed_field 的对比指南 安装 Pydantic不使用 computed_field 的实现特性 使用 computed_field 的实现特性 使用和不使用 computed_field 的对比适用场景分析什么时候不需要 computed_field&#xff1f;什么时候使用 computed_fi…

Docker Engine多平台镜像构建(ARM64、x64、riscv64...)

Docker Engine多平台镜像构建(ARM64、x64、riscv64…) 1. Docker Engine安装 设置 Docker 的存储库# Add Dockers official GPG key: sudo apt-get update sudo apt-get install ca-certificates curl sudo install -m 0755 -d /etc/apt/keyrings sudo curl -fsSL https://do…

连续大涨,汉王科技跑步进入AI应用舒适区

OpenAI正在进行的“12天12场直播”让行业再次沸腾&#xff0c;二级市场也在寻找AI应用的机会。这刺激了12月首周同花顺sora概念涨超11&#xff05;&#xff0c;远超同期大盘指数涨幅。 截至目前&#xff0c;“满血版”推理模型o1和月收费高达200美元的ChatGPT Pro订阅服务&…

[MySQL基础](三)SQL--图形化界面+DML

本专栏内容为&#xff1a;MySQL学习专栏 &#x1f493;博主csdn个人主页&#xff1a;小小unicorn ⏩专栏分类&#xff1a;MySql &#x1f69a;代码仓库&#xff1a;小小unicorn的代码仓库&#x1f69a; &#x1f339;&#x1f339;&#x1f339;关注我带你学习编程知识 目录 图…

基于单片机的智能灯光控制系统

摘要 现在的大部分的大学&#xff0c;都是采用了一种“绿色”的教学方式&#xff0c;再加上现在的大学生缺乏环保意识&#xff0c;所以在学校里很多的教室&#xff0c;在白天的时候灯都会打开&#xff0c;这是一种极大的浪费&#xff0c;而且随时都有可能看到&#xff0c;这是…

数据分析及应用:滴滴出行打车日志数据分析

目录 0 日志数据集介绍 1 构建数据仓库 1.1 ods创建用户打车订单表 1.2 创建分区 1.3 上传到对应分区

解决Windows与Ubuntu云服务器无法通过Socket(udp)通信问题

今天在写Socket通信代码的时候&#xff0c;使用云服务器自己与自己通信没有问题&#xff0c;但是当我们把客户端换为Windows系统的时候却无法发送信息到Linux当中&#xff0c;耗时一上午终于搞定了&#x1f612;。 问题&#xff1a; 如上图&#xff0c;当我在windows的客户端…

网络安全基本命令

网络安全基本命令 想学会网络安全,就必须学会基本的网络常用命令,才能更好的去掌握网络,保护自己的系统&#xff0c;防止入侵。我们必须学会的基本的网络命令主要是基于Windows NT平台下的基本命令&#xff0c;也就是说windows 98/windows ME的下部分命令是不能运行的。所以说&…

帝可得-运营管理App

运营管理App Android模拟器 本项目的App客户端部分已经由前端团队进行开发完成&#xff0c;并且以apk的方式提供出来&#xff0c;供我们测试使用&#xff0c;如果要运行apk&#xff0c;需要先安装安卓的模拟器。 可以选择国内的安卓模拟器产品&#xff0c;比如&#xff1a;网…

Android显示系统(07)- OpenGL ES - 纹理Texture

Android显示系统&#xff08;02&#xff09;- OpenGL ES - 概述 Android显示系统&#xff08;03&#xff09;- OpenGL ES - GLSurfaceView的使用 Android显示系统&#xff08;04&#xff09;- OpenGL ES - Shader绘制三角形 Android显示系统&#xff08;05&#xff09;- OpenGL…

二十万分之一几率:if语句变do-while卡死问题分析

背景 某次灰度发布之后没多久就收到线上ANR告警&#xff0c;经排查定位到是某个页面onCreate方法执行太久导致&#xff0c;而火焰图中的耗时堆栈指向了我们用于监控页面启动速度的一段插桩代码&#xff0c;反编译Apk之后发现本该是if语句的代码竟变成了一个do-while语句&#…

React v19稳定版发布12.5

&#x1f916; 作者简介&#xff1a;水煮白菜王 &#xff0c;一位资深前端劝退师 &#x1f47b; &#x1f440; 文章专栏&#xff1a; 前端专栏 &#xff0c;记录一下平时在博客写作中&#xff0c;总结出的一些开发技巧✍。 感谢支持&#x1f495;&#x1f495;&#x1f495; 目…