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 缺点

部分请求可能未被处理。

相关推荐
2401_8920709820 小时前
【Linux C++ 日志系统实战】LogFile 日志文件管理核心:滚动策略、线程安全与方法全解析
linux·c++·日志系统·日志滚动
yuzhuanhei21 小时前
Visual Studio 配置C++opencv
c++·学习·visual studio
不爱吃炸鸡柳21 小时前
C++ STL list 超详细解析:从接口使用到模拟实现
开发语言·c++·list
十五年专注C++开发21 小时前
RTTR: 一款MIT 协议开源的 C++ 运行时反射库
开发语言·c++·反射
Momentary_SixthSense21 小时前
设计模式之工厂模式
java·开发语言·设计模式
Java码农也是农21 小时前
Multi-Agent 系统设计模式
设计模式·agent·multi-agent
sg_knight21 小时前
设计模式实战:状态模式(State)
python·ui·设计模式·状态模式·state
‎ദ്ദിᵔ.˛.ᵔ₎1 天前
STL 栈 队列
开发语言·c++
2401_892070981 天前
【Linux C++ 日志系统实战】高性能文件写入 AppendFile 核心方法解析
linux·c++·日志系统·文件写对象
郭涤生1 天前
STL vector 扩容机制与自定义内存分配器设计分析
c++·算法