13 责任链(Chain of Responsibility)模式

责任链模式

1.1 分类

(对象)行为型

1.2 提出问题

开发一款故障报修系统,不同的业务员处理的故障不同,如何确保客户的维修请求得到妥善的处理。

1.3 解决方案

使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。

1.4 实现类图

  1. 处理者(Handler)声明了所有具体处理者的通用接口。
  2. 基础处理者(Base Handler)是一个可选的类,你可以将所有处理者共用的样本代码放置在其中。
  3. 具体处理者(Concrete Handlers)包含处理请求的实际代码。 每个处理者接收到请求后,都必须决定是否进行处理,以及是否沿着链传递请求。
  4. 客户端(Client)可根据程序逻辑一次性或者动态地生成链请求可发送给链上的任意一个处理者。

1.5 示例代码

cpp 复制代码
#include <iostream>
#include <string>
#include <list>

//class Handler {
//public:
//    virtual Handler* setNext(Handler* handler) = 0;
//    virtual std::string handle(std::string request) = 0;
//};

class BaseHandler/*:public Handler*/{
private:
    BaseHandler* m_nextHandler;
public:
    BaseHandler* setNext(BaseHandler* handler) {
        m_nextHandler = handler;
        return m_nextHandler;
    }
    virtual std::string handle(std::string request) {
        if (m_nextHandler) {
            return m_nextHandler->handle(request);
        }
        return "";
    }
};

class RobotAI : public BaseHandler {
public:
    std::string handle(std::string request) override {
        if (request == "使用向导") {
            return "RobotAI:我来处理-" + request + "-。\n";
        } else {
            return BaseHandler::handle(request);
        }
    }
};
class TelOperator : public BaseHandler {
public:
    std::string handle(std::string request) override {
        if (request == "常见问题") {
            return "TelOperator:我来处理-" + request + "-。\n";
        } else {
            return BaseHandler::handle(request);
        }
    }
};
class Expert : public BaseHandler {
public:
    std::string handle(std::string request) override {
        if (request == "疑难杂症") {
            return "Expert:我来处理-" + request + "-。\n";
        } else {
            return BaseHandler::handle(request);
        }
    }
};

void clientCode(BaseHandler& handler) {
    std::list<std::string> problems = { "疑难杂症","常见问题","使用向导","常见问题","扯淡的问题" };
    for (const std::string& problem : problems) {
        std::string result = handler.handle(problem);
        std::cout << "Client:谁来处理" << problem<<":\n";
        if (result.empty()) {
            std::cout << "处理结果:" << "没人能够处理。\n";
        } else {
            std::cout << "处理结果:" << result;
        }
    }
}
int main()
{
    RobotAI robot;
    TelOperator telOperator;
    Expert expert;
    robot.setNext(&telOperator)->setNext(&expert);
    std::cout << "Chain:机器人AI->话务员->专家\n";
    clientCode(robot);

    //clientCode(telOperator);
}

1.6 举个栗子

责任链模式负责为活动的 GUI 元素显示上下文帮助信息。

1.7 总结

1.7.1 优点

  1. 你可以控制请求处理的顺序。
  2. 单一职责原则。你可对发起操作和执行操作的类进行解耦。
  3. 开闭原则。你可以在不更改现有代码的情况下在程序中新增处理者。

1.7.2 缺点

部分请求可能未被处理。

相关推荐
郝学胜_神的一滴18 小时前
CMake 034:生成器表达式:解耦构建时序、精简分支逻辑的终极利器
c++·cmake
咖啡八杯1 天前
GoF设计模式——中介者模式
java·后端·spring·设计模式
见过夏天1 天前
C++ 基础入门完全指南
c++
胡萝卜术2 天前
从“分数打架”到“排名投票”:为什么你的ChatBI必须用RRF?
算法·设计模式·面试
亦暖筑序3 天前
Java 8老系统Browser Agent实战:三层拦截把AI操作后台变成可审计流程
java·后端·设计模式
用户805533698033 天前
不止三件套:QObject 属性系统全关键字与运行时反射!
c++·qt
BadBadBad__AK3 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
卷无止境4 天前
Eigen 库如何借助 OpenMP 加速计算
c++·后端
卷无止境4 天前
OpenMPI、MPICH 与 OpenMP:关系、核心概念与架构全解
c++·后端
郝学胜_神的一滴5 天前
CMake 30:循环语法全解|foreach_while双循环精讲、迭代技巧与实战避坑指南
c++·cmake