设计模式-职责链模式

文章目录

职责链模式

职责链模式是一种行为设计模式,它可以将请求的发送者和请求的处理者解耦,并按照预定义的顺序处理请求。职责链模式常用于需要逐级审批或转交处理的场景。

模式概述

职责链模式通过建立一个处理请求的对象链(职责链),将发送者和处理者分离开来。请求沿着职责链依次传递,直到有一个处理者能够处理它为止。这样,请求的发送者不需要知道哪个具体对象会处理请求,而处理者也不需要知道请求的发送者是谁。

主要角色

  • 抽象处理者(Handler):定义了处理请求的接口,并可以设定继任者(successor)。
  • 具体处理者(ConcreteHandler):实现抽象处理者的接口,根据自身的能力判断是否可以处理请求,如果可以则进行处理,否则将请求转发给继任者。

适用场景

适用于以下情况:

  • 系统中存在多个对象可以处理同一请求,但具体处理者在运行时才能确定。
  • 发送者和接收者之间的耦合关系需要降低。
  • 需要动态指定处理链的顺序或配置处理链。

实现步骤

  1. 定义抽象处理者,声明处理请求的方法,并提供设置继任者的方法。
  2. 实现具体处理者,根据自身的能力判断是否能够处理请求,如果可以则进行处理,否则将请求转发给继任者。
  3. 在客户端中创建处理链,并按照预定顺序设置继任者关系。
  4. 将请求发送到处理链的起始点,让处理链逐级处理请求。

优点

  • 解耦了请求的发送者和处理者,增强了系统的灵活性和可维护性。
  • 动态配置处理链,可以灵活地改变请求的处理顺序或增减处理者。
  • 符合单一职责原则和开闭原则。

注意事项

  • 需要正确设置处理链的顺序,确保请求能够被正确处理。
  • 处理链的最后一个处理者必须能够处理所有未被前面的处理者处理的请求,避免请求无法得到处理。

职责链模式可以简化请求的发送和处理过程,提高系统的灵活性和可扩展性。但需要合理设计和配置处理链,以确保请求能够被正确处理。

参考资料:《设计模式:可复用面向对象软件的基础》(GoF)

定义

职责链模式是一种行为设计模式,用于将请求的发送者和请求的处理者解耦,并且能够按照预定义的顺序处理请求。通过职责链模式,处理者负责处理请求,而客户只需将请求发送到职责链上,无需关心请求的处理细节和传递过程。

职责链结构

职责链模式的主要结构包括抽象处理者(Handler)和具体处理者(ConcreteHandler)。抽象处理者定义了一个处理请求的接口,并且可以设定继任者(successor)。具体处理者实现了抽象处理者的接口,并根据自身的能力来判断是否能够处理请求,如果可以处理则进行处理,否则将请求传递给继任者。

示例

以下是一个简单的职责链模式示例的代码:

csharp 复制代码
// 设置处理请求的抽象类
abstract class Handler
{
    protected Handler successor;

    // 设置继任者
    public void SetSuccessor(Handler successor)
    {
        this.successor = successor;
    }

    // 处理请求的抽象方法
    public abstract void HandleRequest(int request);
}

// 具体处理者类
class ConcreteHandler1 : Handler
{
    public override void HandleRequest(int request)
    {
        if (request >= 0 && request < 10)
        {
            Console.WriteLine("{0} 处理请求 {1}", this.GetType().Name, request);
        }
        else if (successor != null)
        {
            // 请求传递给继任者
            successor.HandleRequest(request);
        }
    }
}

// 其他具体处理者类与第一个类似,范围不同

// 客户端代码
static void Main(string[] args)
{
    Handler h1 = new ConcreteHandler1();
    Handler h2 = new ConcreteHandler2();
    Handler h3 = new ConcreteHandler3();

    h1.SetSuccessor(h2); // 设置继任者
    h2.SetSuccessor(h3);

    int[] requests = { 2, 4, 5, 7, 8, 12, 23, 27, 6 };
    foreach (int request in requests)
    {
        h1.HandleRequest(request);
    }

    Console.Read();
}

这段代码是一个简单的职责链模式的实现示例。以下是对这个代码的一些要点解释和改进建议:

  1. 职责链的顺序设置:在当前的代码中,通过SetSuccessor()方法手动设置了处理链的顺序。这种硬编码方式不够灵活,如果需要动态地配置处理链的顺序,可能需要修改代码。可以考虑使用其他方式,比如配置文件或依赖注入,从而实现更灵活的处理链配置。

  2. 请求的范围判断:在具体处理者类中,使用了一个条件判断语句来确定是否能够处理请求。目前的实现中,每个具体处理者都需要实现相同的条件判断逻辑,这样会导致代码重复。可以考虑将请求的范围判断逻辑抽离出来,单独封装成一个工具类或方法,让具体处理者直接调用,避免代码重复。

  3. 继任者的处理:当前的代码中,当具体处理者无法处理请求时,会将请求传递给继任者。这种方式可以确保请求能够被处理,但可能存在一个问题:如果继任者也无法处理请求,请求会一直传递下去直到最后一个处理者。为了避免请求无限传递,可以考虑在最后一个处理者中添加一个默认的处理逻辑,比如抛出一个异常或给出一个默认的处理结果。

  4. 扩展性和灵活性:职责链模式的一个重要特点是扩展性和灵活性。当前的代码通过继承和设置继任者的方式实现了处理链,这样可以方便地添加新的处理者并灵活调整处理顺序。但在实际应用中,可能还需要考虑更复杂的场景,比如动态添加、删除和调整处理者等。可以根据具体需求进行扩展和改进。

  5. 错误处理和异常处理:在当前的代码中,并没有专门处理处理请求时可能出现的错误或异常情况。为了提高代码的健壮性,可以在处理请求时添加适当的错误处理和异常处理机制,以确保系统的稳定性和可靠性。

在上述示例中,根据请求的大小范围,具体处理者对象判断是否可以处理该请求。如果可以处理,则进行处理;如果不能处理,则将请求传递给继任者进行处理。这样的处理过程可以按照设置好的顺序依次执行。

总结

职责链模式的优点包括解耦请求发送者和处理者、动态配置处理链、增强系统的灵活性和可扩展性等。但需要注意的是,如果没有正确配置处理链或最后一个处理者无法处理请求,请求可能无法得到处理。

相关推荐
serve the people2 分钟前
springboot 单独新建一个文件实时写数据,当文件大于100M时按照日期时间做文件名进行归档
java·spring boot·后端
qmx_071 小时前
HTB-Jerry(tomcat war文件、msfvenom)
java·web安全·网络安全·tomcat
为风而战1 小时前
IIS+Ngnix+Tomcat 部署网站 用IIS实现反向代理
java·tomcat
编程零零七2 小时前
Python数据分析工具(三):pymssql的用法
开发语言·前端·数据库·python·oracle·数据分析·pymssql
技术无疆3 小时前
快速开发与维护:探索 AndroidAnnotations
android·java·android studio·android-studio·androidx·代码注入
2401_858286113 小时前
52.【C语言】 字符函数和字符串函数(strcat函数)
c语言·开发语言
铁松溜达py3 小时前
编译器/工具链环境:GCC vs LLVM/Clang,MSVCRT vs UCRT
开发语言·网络
everyStudy3 小时前
JavaScript如何判断输入的是空格
开发语言·javascript·ecmascript
C-SDN花园GGbond5 小时前
【探索数据结构与算法】插入排序:原理、实现与分析(图文详解)
c语言·开发语言·数据结构·排序算法
迷迭所归处6 小时前
C++ —— 关于vector
开发语言·c++·算法