【知识科普】设计模式之-责任链模式

这里写自定义目录标题

概述

责任链模式的详细描述

责任链模式(Chain of Responsibility Pattern)是一种行为型设计模式,它允许多个对象按照一定顺序处理请求,并且每个对象都有机会选择是否处理该请求或将其传递给下一个对象。这种模式将请求的发送者和接收者解耦,提供了更大的灵活性和可扩展性。

在责任链模式中,有两个主要角色:

  1. 抽象处理者(Handler):定义了处理请求的接口,并持有对下一个处理者的引用。
  2. 具体处理者(ConcreteHandler):实现了抽象处理者接口,负责实际处理请求的逻辑。

当一个请求从链首开始传递时,每个处理者都有机会处理该请求,直到其中一个处理者能够处理为止。如果当前处理者不能处理该请求,它会将请求传递给下一个处理者。这样,请求发送者无需知道具体的处理对象,只需将请求发送到责任链上即可。

责任链模式的使用场景

责任链模式通常用于以下场景:

  1. 多个对象可以处理同一请求:当系统中存在多个对象可以处理同一请求时,可以使用责任链模式将这些对象组织成一条链,并动态地选择处理者。
  2. 处理者之间的顺序不确定:当处理请求的对象之间的顺序不确定或需要动态调整时,责任链模式可以灵活地组织处理者的顺序。
  3. 请求处理具有多种可能性:当请求的处理结果有多种可能性,且这些可能性需要在运行时确定时,责任链模式可以根据请求的内容和当前处理者的逻辑来决定如何处理请求。

使用场景举例

责任链模式的使用场景通常涉及多个对象可以处理同一请求,且这些对象之间的处理顺序或责任分配可能在运行时动态变化的情况。以下是一些具体的使用场景示例:

1. 审批流程

在一个企业中,可能有多个层级的审批人员(如部门经理、总监、副总裁、总裁等)需要审批某个项目或请求。每个审批人员都有自己的审批权限和规则。使用责任链模式,可以将这些审批人员组织成一条链,当请求提交时,它会依次传递给每个审批人员,直到找到能够批准该请求的人员为止。

示例:

  • 抽象审批者:定义了审批请求的方法。
  • 具体审批者(如部门经理、总监等):实现了抽象审批者的方法,并根据自己的权限和规则决定是否批准请求或传递给下一个审批者。

2. 过滤器链

在Web开发中,经常需要对请求和响应进行过滤,如身份验证、日志记录、输入验证等。使用责任链模式,可以将这些过滤器组织成一条链,每个过滤器都有机会处理请求或响应,并在必要时将控制权传递给下一个过滤器。

示例:

  • 抽象过滤器:定义了处理请求和响应的方法。
  • 具体过滤器(如身份验证过滤器、日志记录过滤器等):实现了抽象过滤器的方法,并根据自己的逻辑处理请求或响应。

3. 事件处理系统

在一些应用程序中,可能需要处理各种类型的事件,如用户点击、键盘输入、系统错误等。使用责任链模式,可以将这些事件的处理程序组织成一条链,每个处理程序都有机会处理事件,并在必要时将事件传递给下一个处理程序。

示例:

  • 抽象事件处理器:定义了处理事件的方法。
  • 具体事件处理器(如用户点击处理器、键盘输入处理器等):实现了抽象事件处理器的方法,并根据事件类型和自己的逻辑处理事件。

4. 插件系统

在一些软件框架或平台中,可能允许用户或开发者添加自定义的插件来扩展功能。使用责任链模式,可以将这些插件组织成一条链,并在运行时动态地加载和调用它们。每个插件都有机会处理请求或提供自己的功能,并在必要时将控制权传递给下一个插件。

示例:

  • 抽象插件:定义了插件需要实现的方法。
  • 具体插件:实现了抽象插件的方法,并提供了自定义的功能或处理逻辑。

这些场景展示了责任链模式在动态责任分配、请求处理和事件处理方面的灵活性。通过组织成链式的处理者,责任链模式允许系统在运行时动态地决定由哪个对象处理请求,从而提高了系统的可扩展性和可维护性。

Java代码示例及注释

以下是一个使用Java实现责任链模式的示例代码,以及对应的注释:

java 复制代码
// 定义请求类
class Request {
    private String content;

    public Request(String content) {
        this.content = content;
    }

    public String getContent() {
        return content;
    }
}

// 定义抽象处理者接口
interface Handler {
    void handleRequest(Request request);
}

// 具体处理者A,实现Handler接口
class ConcreteHandlerA implements Handler {
    private Handler successor;

    public void setSuccessor(Handler successor) {
        this.successor = successor;
    }

    @Override
    public void handleRequest(Request request) {
        if (request.getContent().contains("A")) {
            System.out.println("ConcreteHandlerA: Handling the request");
        } else if (successor != null) {
            successor.handleRequest(request);
        }
    }
}

// 具体处理者B,实现Handler接口
class ConcreteHandlerB implements Handler {
    private Handler successor;

    public void setSuccessor(Handler successor) {
        this.successor = successor;
    }

    @Override
    public void handleRequest(Request request) {
        if (request.getContent().contains("B")) {
            System.out.println("ConcreteHandlerB: Handling the request");
        } else if (successor != null) {
            successor.handleRequest(request);
        }
    }
}

// 客户端代码
public class Main {
    public static void main(String[] args) {
        // 创建处理者对象
        Handler handlerA = new ConcreteHandlerA();
        Handler handlerB = new ConcreteHandlerB();

        // 设置处理者链
        handlerA.setSuccessor(handlerB);

        // 创建请求对象
        Request request1 = new Request("This is requestA");
        Request request2 = new Request("This is requestB");
        Request request3 = new Request("This is requestC");

        // 处理请求
        handlerA.handleRequest(request1); // ConcreteHandlerA: Handling the request
        handlerA.handleRequest(request2); // ConcreteHandlerB: Handling the request
        handlerA.handleRequest(request3); // 无输出,因为两个处理者都不处理requestC
    }
}

代码解释

  1. Request类:表示一个请求,包含请求的内容。
  2. Handler接口:定义了处理请求的接口,所有具体处理者都需要实现这个接口。
  3. ConcreteHandlerA和ConcreteHandlerB类:实现了Handler接口,分别处理包含"A"和"B"的请求内容。如果请求内容不匹配,它们会将请求传递给下一个处理者。
  4. Main类:客户端代码,创建处理者对象并设置处理者链,然后创建请求对象并调用处理请求的方法。

在这个示例中,当发送一个包含"A"的请求时,ConcreteHandlerA会处理该请求;当发送一个包含"B"的请求时,ConcreteHandlerB会处理该请求;如果发送一个包含"C"的请求,则两个处理者都不会处理该请求,因为没有处理者包含处理"C"的逻辑。

相关推荐
天乐敲代码3 分钟前
JAVASE入门十五脚-网络TCP,UDP,,Lambda
java
2501_903238651 小时前
自定义登录页面的Spring Security实践
java·后端·spring·个人开发
飞翔的佩奇2 小时前
Java项目: 基于SpringBoot+mybatis+maven+mysql实现的图书管理系统(含源码+数据库+答辩PPT+毕业论文)
java·数据库·spring boot·mysql·spring·毕业设计·图书管理
jerry6094 小时前
注解(Annotation)
java·数据库·sql
Future_yzx4 小时前
Java Web的发展史与SpringMVC入门学习(SpringMVC框架入门案例)
java·前端·学习
辞半夏丶北笙5 小时前
最近最少使用算法(LRU最近最少使用)缓存替换算法
java·算法·缓存
星如雨グッ!(๑•̀ㅂ•́)و✧5 小时前
Java NIO全面详解
java·python·nio
taopi20247 小时前
android java系统弹窗的基础模板
android·java·开发语言
松仔log7 小时前
Java多线程——对象的组合
java·开发语言·jvm
酷爱码7 小时前
springboot 动态配置定时任务
java·spring boot·后端