Qt C++设计模式->责任链模式

责任链模式(Chain of Responsibility Pattern)是一种行为型设计模式,它允许多个对象有机会处理请求,而不需要明确指定哪个对象处理。通过将这些对象连成一条链,请求沿着链传递,直到有对象处理它为止。该模式的主要目标是使请求的发送者与接收者解耦,且能够动态调整处理链。

责任链模式的应用场景

责任链模式适用于当有多个对象可以处理一个请求,但具体由哪个对象处理在运行时才能确定的情况。它可以避免发送者和处理者之间的紧耦合。常见的应用场景包括:

  • 事件处理机制:例如GUI中的事件分发,某个控件未处理时,会传递给上层控件。

  • 日志处理:日志请求可以通过不同的级别(如DEBUG、INFO、ERROR)进行处理,责任链可以动态调整处理链。

责任链模式的核心

  • 处理者(Handler):每个处理者对象都包含对下一个处理者的引用,请求从第一个处理者开始,沿着链条依次传递。

  • 请求(Request):可以是客户端发出的任何类型的请求,如事件、命令或任务。

每个处理者都有两个选择:

  1. 处理请求。

  2. 将请求传递给下一个处理者。

责任链模式示例代码

假设你正在开发一个简单的日志系统,日志可以按照不同的级别(例如DEBUG、INFO、ERROR)进行处理。不同的日志级别可能需要不同的处理方式。

cpp 复制代码
#include <QDebug>
#include <QString>

// 抽象处理者类:定义日志处理接口
class Logger {
protected:
    Logger* next;  // 下一个处理者

public:
    Logger() : next(nullptr) {}

    void setNext(Logger* nextLogger) {
        next = nextLogger;  // 设置下一个处理者
    }

    // 处理日志请求的方法
    void logMessage(const QString& level, const QString& message) {
        if (canHandle(level)) {
            handle(message);  // 当前处理者可以处理该请求
        } else if (next) {
            next->logMessage(level, message);  // 传递给下一个处理者
        } else {
            qDebug() << "Unhandled log level:" << level;
        }
    }

    virtual ~Logger() = default;

protected:
    virtual bool canHandle(const QString& level) const = 0;  // 判断当前处理者是否能处理该请求
    virtual void handle(const QString& message) const = 0;    // 处理请求
};

// 具体处理者类:处理DEBUG级别的日志
class DebugLogger : public Logger {
protected:
    bool canHandle(const QString& level) const override {
        return level == "DEBUG";  // 处理DEBUG日志
    }

    void handle(const QString& message) const override {
        qDebug() << "[DEBUG]:" << message;
    }
};

// 具体处理者类:处理INFO级别的日志
class InfoLogger : public Logger {
protected:
    bool canHandle(const QString& level) const override {
        return level == "INFO";  // 处理INFO日志
    }

    void handle(const QString& message) const override {
        qDebug() << "[INFO]:" << message;
    }
};

// 具体处理者类:处理ERROR级别的日志
class ErrorLogger : public Logger {
protected:
    bool canHandle(const QString& level) const override {
        return level == "ERROR";  // 处理ERROR日志
    }

    void handle(const QString& message) const override {
        qDebug() << "[ERROR]:" << message;
    }
};

// 使用示例
int main() {
    // 创建具体处理者
    Logger* debugLogger = new DebugLogger();
    Logger* infoLogger = new InfoLogger();
    Logger* errorLogger = new ErrorLogger();

    // 设置责任链顺序:DEBUG -> INFO -> ERROR
    debugLogger->setNext(infoLogger);
    infoLogger->setNext(errorLogger);

    // 发送不同级别的日志请求
    debugLogger->logMessage("DEBUG", "This is a debug message.");
    debugLogger->logMessage("INFO", "This is an info message.");
    debugLogger->logMessage("ERROR", "This is an error message.");
    debugLogger->logMessage("TRACE", "This is a trace message.");  // 未处理的日志级别

    // 清理内存
    delete debugLogger;
    delete infoLogger;
    delete errorLogger;

    return 0;
}

代码解析

  • Logger类 :抽象处理者类,定义了处理日志请求的接口,并包含指向下一个处理者的指针。logMessage方法判断是否由当前处理者处理请求,如果不能处理,则传递给下一个处理者。

  • DebugLogger、InfoLogger、ErrorLogger类 :具体处理者类,分别处理不同级别的日志(DEBUG、INFO、ERROR)。每个类都重写了canHandlehandle方法,根据日志级别处理相应的请求。

  • 客户端代码:客户端通过责任链发送不同级别的日志请求,链中的每个处理者依次判断是否处理该请求,未处理则传递给下一个处理者。

责任链模式的优点

  • 解耦请求发送者与处理者:请求发送者不需要知道具体哪个处理者会处理请求,减少了对象之间的耦合。

  • 灵活的请求处理链:可以在运行时动态修改处理链的结构,比如在链中添加或移除处理者,或者改变处理者的顺序。

  • 职责分离:每个处理者只关注处理自己能处理的请求,其余的请求交给下一个处理者处理,实现了职责的分离。

责任链模式的缺点

  • 可能导致请求处理无响应:如果没有合适的处理者能够处理请求,可能会导致请求被忽略。为了避免这一问题,可以在链的末端添加一个默认处理者,来处理所有未被处理的请求。

  • 性能开销:责任链的请求会被多个处理者依次传递,可能会导致链条过长,从而影响性能。

适合使用责任链模式的情况

  • 需要动态定义请求的处理顺序:在需要根据运行时条件动态定义处理顺序时,责任链模式非常适合。

  • 多个对象可以处理相同的请求:当多个对象可以处理同一类请求,但具体由哪个对象处理在运行时才确定时,使用责任链模式可以避免硬编码多个if-else语句。

  • 希望减少请求发送者与处理者的耦合:责任链模式能够很好的解耦请求的发送者与请求的处理者,增加系统的灵活性。

Qt中的责任链模式应用

在Qt开发中,责任链模式可以用于事件处理机制。例如,Qt中的事件分发机制(QEvent)中,事件从最底层的组件传递到父组件,直到有组件处理该事件为止,这就是责任链模式的典型应用。此外,日志系统、命令处理等场景也可以通过责任链模式实现灵活的请求处理。

责任链模式能够很好地解耦请求发送者与处理者,同时提供了动态调整请求处理流程的能力,适合用于多个对象能够处理请求的场景。

相关推荐
芊寻(嵌入式)6 分钟前
C转C++学习笔记--基础知识摘录总结
开发语言·c++·笔记·学习
WaaTong7 分钟前
《重学Java设计模式》之 原型模式
java·设计模式·原型模式
獨枭8 分钟前
C++ 项目中使用 .dll 和 .def 文件的操作指南
c++
霁月风11 分钟前
设计模式——观察者模式
c++·观察者模式·设计模式
橘色的喵11 分钟前
C++编程:避免因编译优化引发的多线程死锁问题
c++·多线程·memory·死锁·内存屏障·内存栅栏·memory barrier
何曾参静谧1 小时前
「C/C++」C/C++ 之 变量作用域详解
c语言·开发语言·c++
AI街潜水的八角1 小时前
基于C++的决策树C4.5机器学习算法(不调包)
c++·算法·决策树·机器学习
JSU_曾是此间年少1 小时前
数据结构——线性表与链表
数据结构·c++·算法
此生只爱蛋2 小时前
【手撕排序2】快速排序
c语言·c++·算法·排序算法
何曾参静谧3 小时前
「C/C++」C/C++ 指针篇 之 指针运算
c语言·开发语言·c++