如何在业务代码中优雅使用责任链模式

news2025/1/22 13:15:13

责任链模式(Chain of Responsibility Pattern)为请求创建了一个接收者对象的链。这种模式给予请求的类型,对请求的发送者和接收者进行解耦。这种类型的设计模式属于行为型模式。

在这种模式中,通常每个接收者都包含对另一个接收者的引用。如果一个对象不能处理该请求,那么它会把相同的请求传给下一个接收者,依此类推。

图片

image.png

责任链模式介绍

意图: 避免请求发送者与接收者耦合在一起,让多个对象都有可能接收请求,将这些对象连接成一条链,并且沿着这条链传递请求,直到有对象处理它为止。

主要解决: 职责链上的处理者负责处理请求,客户只需要将请求发送到职责链上即可,无须关心请求的处理细节和请求的传递,所以职责链将请求的发送者和请求的处理者解耦了。

何时使用: 在处理消息的时候以过滤很多道。

如何解决: 拦截的类都实现统一接口。

关键代码: Handler 里面聚合它自己,在 HandlerRequest 里判断是否合适,如果没达到条件则向下传递,向谁传递之前 set 进去。

应用实例:  1、红楼梦中的"击鼓传花"。2、JS 中的事件冒泡。3、JAVA WEB 中 Apache Tomcat 对 Encoding 的处理,Struts2 的拦截器,jsp servlet 的 Filter。

优点:  1、降低耦合度。它将请求的发送者和接收者解耦。2、简化了对象。使得对象不需要知道链的结构。3、增强给对象指派职责的灵活性。通过改变链内的成员或者调动它们的次序,允许动态地新增或者删除责任。4、增加新的请求处理类很方便。

缺点:  1、不能保证请求一定被接收。2、系统性能将受到一定影响,而且在进行代码调试时不太方便,可能会造成循环调用。3、可能不容易观察运行时的特征,有碍于除错

业务代码使用的例子

利用spring boot注入及接口实现责任链

首先我们定义一个入参Payment

public class Payment {  
    private boolean success;  

    // 其他参数略....

    public boolean isSuccess() {  
        return success;  
    }  

    public void setSuccess(boolean success) {  
        this.success = success;  
    }  
}

再定义一个接口

public interface PaymentProcessor {  
    /**  
    * 节点处理  
    *  
    * @param context  
    */  
    void handle(Payment context);  

}

接下来我们定义两个实现类CreditCardProcessorPayPalProcessor.当我们新增节点或者实现时可以直接实现PaymentProcessor接口。这里采用spring注解@Order来定义执行顺序。

@Order(1)  
@Component  
public class CreditCardProcessor implements PaymentProcessor {  
    @Override  
    public void handle(Payment context) {  
        System.out.println("Processed credit card payment.");  
    }  
}


@Order(2)  
@Component  
public class PayPalProcessor implements PaymentProcessor {  
    @Override  
    public void handle(Payment context) {  
        System.out.println("Processed PayPal payment.");  

    }  
}

最后,我们还需要创建一个支付处理servicePaymentHandleChainService,用于管理这些实现类。·这里采用spring注入list的形式,list顺序为上面实现类@Order的顺序

@Service  
public class PaymentHandleChainService {  
    @Autowired  
    private List<PaymentProcessor> paymentProcessors;  

    public void execute(Payment payment) {  
        for (PaymentProcessor paymentProcessor : paymentProcessors) {  
            paymentProcessor.handle(payment);  
        }  
    }  
}

整体结构如下图所示:

图片

PaymentHandleChainService.png

我们写个单元测试:

@RunWith(SpringRunner.class)  
@SpringBootTest(classes = SpringExampleApplication.class)  
public class PaymentServiceTest {  
    @Autowired  
    private PaymentHandleChainService paymentHandleChainService;  

    @Test  
    public void test() {  
        paymentHandleChainService.execute(new Payment());  
    }  
}

结果如下图所示,符合我们预期:

抽象类实现责任链

另一种方式通过抽象类定义链式,我们还是用上面的例子,这里加个抽象类。

public abstract class AbstractPaymentProcessor {  
    /**  
    * 下一个节点  
    */  
    protected AbstractPaymentProcessor next = null;  

    public void execute(Payment context) throws Exception {  
        // 上层未执行成功,不再执行  
        if (!context.isSuccess()) {  
            return;  
        }  
        // 执行当前阶段  
        doHandler(context);  
        // 判断是否还有下个责任链节点,没有的话,说明已经是最后一个节点  
        if (getNext() != null) {  
            getNext().execute(context);  
        }  
    }  

    public AbstractPaymentProcessor getNext() {  
        return next;  
    }  

    public void setNext(AbstractPaymentProcessor next) {  
        this.next = next;  
    }  

    public abstract void doHandler(Payment content) throws Exception;  

    public static class Builder {  
        private AbstractPaymentProcessor head;  
        private AbstractPaymentProcessor tail;  

        public Builder addHandler(AbstractPaymentProcessor handler) {  
            if (this.head == null) {  
                this.head = handler;  
            } else {  
                this.tail.setNext(handler);  
            }  
            this.tail = handler;  
            return this;  
        }  

        public AbstractPaymentProcessor build() {  
            return this.head;  
        }  
    }  
}

新定义两个实现类CreditCard2ProcessorPayPal2Processor

@Component  
public class CreditCard2Processor extends AbstractPaymentProcessor {  

    @Override  
    public void doHandler(Payment content) throws Exception {  
        System.out.println("Processed credit card payment.");  
    }  
}

@Component  
public class PayPal2Processor extends AbstractPaymentProcessor {  

    @Override  
    public void doHandler(Payment content) throws Exception {  
        System.out.println("Processed PayPal payment.");  
    }  
}

这种方式使用起来可以自定义节点,比较灵活。

@Test  
public void test2() throws Exception {  
    paymentHandleChainService.execute(new Payment());  
    new AbstractPaymentProcessor.Builder()  
        .addHandler(creditCard2Processor)  
        .addHandler(payPal2Processor)  
        .build().execute(new Payment());  
}

整体结构如下:

责任链模式与策略模式区别

之前我们讲过如何在业务代码中优雅的使用策略模式,下面我们来看一下两者的区别。

责任链模式和策略模式都是常见的行为型设计模式,但它们解决的问题和应用场景有一些不同之处。以下是责任链模式和策略模式的主要区别:

  1. 问题域不同:

    • 责任链模式(Chain of Responsibility):用于构建一个由多个处理器组成的处理链,每个处理器依次尝试处理请求,直到请求被处理或链上没有处理器能够处理为止。主要用于分离请求发送者和接收者,避免紧耦合的处理方式。

    • 策略模式(Strategy):用于定义一组算法或行为,使它们可以相互替换。主要用于在运行时根据不同的情况选择不同的策略,从而实现不同的行为。

  2. 关注点不同:

    • 责任链模式:关注的是请求的处理流程,它将多个处理器连接起来形成一个处理链,每个处理器负责处理一部分请求,或者将请求传递给下一个处理器。

    • 策略模式:关注的是算法的选择和替换,它将不同的算法封装成策略对象,然后在运行时根据需要选择合适的策略来执行。

  3. 调用顺序不同:

    • 责任链模式:请求会依次在处理链上传递,每个处理器决定是否处理该请求或将其传递给下一个处理器。

    • 策略模式:客户端代码选择合适的策略对象,然后直接调用所选策略的方法。

  4. 目的不同:

    • 责任链模式:主要用于处理请求的分发和处理,可以用于动态地组织和调整处理器的顺序和层次。

    • 策略模式:主要用于实现不同的算法或行为,使客户端代码能够根据需求选择适当的策略来完成任务。

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

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

相关文章

LLM:RoPE位置编码

论文&#xff1a;https://arxiv.org/pdf/2104.09864.pdf 代码&#xff1a;https://github.com/ZhuiyiTechnology/roformer 发表&#xff1a;2021 绝对位置编码&#xff1a;其常规做法是将位置信息直接加入到输入中&#xff08;在x中注入绝对位置信息&#xff09;。即在计算 q…

mysql原理--锁

1.解决并发事务带来问题的两种基本方式 上一章唠叨了事务并发执行时可能带来的各种问题&#xff0c;并发事务访问相同记录的情况大致可以划分为3种&#xff1a; (1). 读-读 情况&#xff1a;即并发事务相继读取相同的记录。 读取操作本身不会对记录有一毛钱影响&#xff0c;并不…

爬虫笔记(一):实战登录古诗文网站

需求&#xff1a;登录古诗文网站&#xff0c;账号&#xff0b;密码&#xff0b;图形验证码 第一&#xff1a;自己注册一个账号&#xff0b;密码哈 第二&#xff1a;图形验证码&#xff0c;需要一个打码平台&#xff08;充钱&#xff0c;超能力power&#xff01;&#xff09;或…

“GPC爬虫池有用吗?

作为光算科技的独有技术&#xff0c;在深入研究谷歌爬虫推出的一种吸引谷歌爬虫的手段 要知道GPC爬虫池是否有用&#xff0c;就要知道谷歌爬虫这一概念&#xff0c;谷歌作为一个搜索引擎&#xff0c;里面有成百上千亿个网站&#xff0c;对于里面的网站内容&#xff0c;自然不可…

网络爬虫采集工具

在当今数字化的时代&#xff0c;获取海量数据对于企业、学术界和个人都至关重要。网络爬虫成为一种强大的工具&#xff0c;能够从互联网上抓取并提取所需的信息。本文将专心分享关于网络爬虫采集数据的全面指南&#xff0c;深入探讨其原理、应用场景以及使用过程中可能遇到的挑…

php array_diff 比较两个数组bug避坑 深入了解

今天实用array_diff出现的异常问题&#xff0c;预想的结果应该是返回 "integral_initiate">"0"&#xff0c;实际没有 先看测试代码&#xff1a; $a ["user_name">"测","see_num">0,"integral_initiate&quo…

leetCode-42.接雨水

&#x1f4d1;前言 本文主要是【算法】——算法模拟的文章&#xff0c;如果有什么需要改进的地方还请大佬指出⛺️ &#x1f3ac;作者简介&#xff1a;大家好&#xff0c;我是听风与他&#x1f947; ☁️博客首页&#xff1a;CSDN主页听风与他 &#x1f304;每日一句&#xff…

Go语言基础快速上手

1、Go语言关键字 2、Go数据类型 3、特殊的操作 3.1、iota关键字 Go中没有明确意思上的enum&#xff08;枚举&#xff09;定义&#xff0c;不过可以借用iota标识符实现一组自增常亮值来实现枚举类型。 const (a iota // 0b // 1c 100 // 100d // 100 (与上一…

H - Least Common Multiple H - 最小公倍数

题目 The least common multiple (LCM) of a set of positive integers is the smallest positive integer which is divisible by all the numbers in the set. For example, the LCM of 5, 7 and 15 is 105. 一组正整数的最小公倍数 &#xff08;LCM&#xff09; 是最小的正整…

rabbitmq的介绍、使用、案例

1.介绍 rabbitmq简单来说就是个消息中间件&#xff0c;可以让不同的应用程序之间进行异步的通信&#xff0c;通过消息传递来实现解耦和分布式处理。 消息队列&#xff1a;允许将消息发到队列&#xff0c;然后进行取出、处理等操作&#xff0c;使得生产者和消费者之间能够解耦&…

使用 Kali Linux Hydra 工具进行攻击测试和警报生成

一、Hydra 工具和 Kali Linux 简介 在网络安全领域中&#xff0c;渗透测试是评估系统密码强度的重要组成部分。Hydra 是一款由黑客组织“The Hackers Choice”开发的开源登录破解工具&#xff0c;支持50多种协议。本教程将探索如何将 Hydra 与 Kali Linux 结合使用&#xff0c…

快快销ShopMatrix 分销商城多端uniapp可编译5端 - 升级申请(可自定义申请表单)

在企业或组织中&#xff0c;升级申请通常涉及到员工职位、权限、设备或者其他资源的提升或更新。创建一个可自定义的升级申请表单可以帮助更高效地收集和处理这类申请信息。以下是一个基本的步骤&#xff1a; 确定表单字段&#xff1a; 申请人信息&#xff1a;姓名、部门、职位…

【Spring Boot 3】【Redis】基本数据类型操作

【Spring Boot 3】【Redis】基本数据类型操作 背景介绍开发环境开发步骤及源码工程目录结构 背景 软件开发是一门实践性科学&#xff0c;对大多数人来说&#xff0c;学习一种新技术不是一开始就去深究其原理&#xff0c;而是先从做出一个可工作的DEMO入手。但在我个人学习和工…

priority_queue的使用与模拟实现(容器适配器+stack与queue的模拟实现源码)

priority_queue的使用与模拟实现 引言&#xff08;容器适配器&#xff09;priority_queue的介绍与使用priority_queue介绍接口使用默认成员函数 size与emptytoppush与pop priority_queue的模拟实现构造函数size与emptytoppush与pop向上调整建堆与向下调整建堆向上调整建堆向下调…

UE5 蓝图编辑美化学习

虚幻引擎中干净整洁蓝图的15个提示_哔哩哔哩_bilibili 1.双击线段成节点。 好用&#xff0c;爱用 2.用序列节点 好用&#xff0c;爱用 3.用枚举。 好用&#xff0c;能避免一些的拼写错误 4.对齐节点 两点一水平线 5.节点上下贴节点 &#xff08;以前不懂&#xff0c;现在经常…

【AJAX框架】AJAX入门与axios的使用

文章目录 前言一、AJAX是干什么的&#xff1f;二、AJAX的安装2.1 CDN引入2.2 npm安装 三、基础使用3.1 CDN方式3.2 node方式 总结 前言 在现代Web开发中&#xff0c;异步JavaScript和XML&#xff08;AJAX&#xff09;已经成为不可或缺的技术之一。AJAX使得网页能够在不刷新整个…

SQL注入实战操作

一&#xff1a;SQl注入分类 按照注入的网页功能类型分类&#xff1a; 1、登入注入&#xff1a;表单&#xff0c;如登入表单&#xff0c;注册表单 2、cms注入&#xff1a;CMS逻辑:index.php首页展示内容&#xff0c;具有文章列表(链接具有文章id)、articles.php文 章详细页&a…

NX二次开发封装自己的函数及如何导入工程

目录 一、概述 二、函数封装 三、函数引用 四、案例——在NX中运行后输出“测试”两字 一、概述 随着对NX二次开发的学习&#xff0c;我们在各种项目里面会积累很多函数&#xff0c;对于一些经常用到的函数&#xff0c;我们可以考虑将其封装为类库&#xff0c;以后在开发其…

从请购到结算,轻松搞定!云迈ERP系统助力企业采购管理全流程!

​在企业的运营过程中&#xff0c;采购管理是至关重要的环节之一。为了确保采购流程的顺畅和高效&#xff0c;许多企业选择引入ERP&#xff08;企业资源规划&#xff09;系统来进行采购管理。那么erp系统中的采购管理都有哪些功能呢&#xff1f; 云迈erp系统中的采购管理模块&a…

虚拟机下载docker

一&#xff0c;Docker简介 百科说&#xff1a;Docker 是一个开源的应用容器引擎&#xff0c;让开发者可以打包他们的应用以及依赖包到一个可移植的容器中&#xff0c;然后发布到任何流行的Linux机器上&#xff0c;也可以实现虚拟化&#xff0c;容器是完全使用沙箱机制&#xff…