当前位置: 首页 > 编程笔记 >

C++设计模式之职责链模式

公冶浩慨
2023-03-14
本文向大家介绍C++设计模式之职责链模式,包括了C++设计模式之职责链模式的使用技巧和注意事项,需要的朋友参考一下

前言

最近心情很差,因为生活,因为工作;所以想请几天假去丽江玩玩。就向项目经理提交了休假申请,我的项目经理向项目主管提交了我的休假申请,项目主管向部门经理提交了我的休假申请;最后,部门经理同意了我的休假申请。是的,一个简单的休假申请,需要这么复杂的流程,这也是一个公司保证它正常运行的必要。如果部门经理休假了,那么我的休假申请由谁审批呢?这个时候由项目主管代替部门经理进行审批。一个休假申请的审批制度有着严格的要求。而在处理这个请假审批时,各个人员就好比在一条链上的节点,我不知道我的请求由谁审批,但是,我的请求最终会有人来处理的。而这样的一种行为,就好比我今天需要总结的职责链模式一样。

什么是职责链模式?

在GOF的《设计模式:可复用面向对象软件的基础》一书中对职责链模式是这样说的:使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这些对象连成一条链,直到有一个对象处理它为止;如下图:

对于每个角色,他们都有他们的职责;当我提交了休假申请时,项目经理需要判断,看看自己能否处理,如果休假超过了2个小时,那么项目经理就不能处理了;项目经理将这个请求提交到项目主管,项目主管判断部门经理在不在,如果部门经理在,项目主管就不能处理了;最后,我的休假申请就到了部门经理那里了,由他亲自审批。可以很明显的看到,项目经理、项目主管和部门经理都有可能处理我的休假申请,我的请求沿着这条链一直走下去,直到有人处理了我的请求。

UML类图

Handler:定义了一个处理请求的接口;其它类如果需要处理相同的请求,可以实现该接口就好了;
ConcreteHandler:处理它所负责的请求,如果可处理该请求,就处理掉这个请求;否则将该请求转发给它的下一个可以处理该请求的对象,所以它必须能访问它的下一个可以处理同样请求的对象;
Client:向处理对象提出具体的请求。

当客户提交一个请求时,请求沿着一条链传递,直至有一个ConcreteHandler对象负责处理它。

使用场合

1.有多个的对象可以处理一个请求,由哪个对象处理该请求是在运行时刻自动确定的;
2.如果想在不明确指定接收者的情况下,向多个对象中的一个提交一个请求;
3.可以处理一个请求的对象集合应被动态指定。

代码实现


#include <iostream>

using namespace std;

 

#define SAFE_DELETE(p) if (p) { delete p; p = NULL; }

 

class HolidayRequest

{

public:

     HolidayRequest(int hour) : m_iHour(hour){}

     int GetHour() { return m_iHour; }

private:

     int m_iHour;

};

 

// The holiday request handler interface

class Manager

{

public:

     virtual bool HandleRequest(HolidayRequest *pRequest) = 0;

};

 

// Project manager

class PM : public Manager

{

public:

     PM(Manager *handler) : m_pHandler(handler){}

     bool HandleRequest(HolidayRequest *pRequest)

     {

          if (pRequest->GetHour() <= 2 || m_pHandler == NULL)

          {

               cout<<"PM said:OK."<<endl;

               return true;

          }

          return m_pHandler->HandleRequest(pRequest);

     }

private:

     Manager *m_pHandler;

};

 

// Department manager

class DM : public Manager

{

public:

     DM(Manager *handler) : m_pHandler(handler){}

     bool HandleRequest(HolidayRequest *pRequest)

     {

          cout<<"DM said:OK."<<endl;

          return true;

     }

 

     // The department manager is in?

     bool IsIn()

     {

          return true;

     }

private:

     Manager *m_pHandler;

};

 

// Project supervisor

class PS : public Manager

{

public:

     PS(Manager *handler) : m_pHandler(handler){}

     bool HandleRequest(HolidayRequest *pRequest)

     {

          DM *pDM = dynamic_cast<DM *>(m_pHandler);

          if (pDM != NULL)

          {

               if (pDM->IsIn())

               {

                    return pDM->HandleRequest(pRequest);

               }

          }

          cout<<"PS said:OK."<<endl;

          return true;

     }

private:

     Manager *m_pHandler;

};

int main()

{

     DM *pDM = new DM(NULL);

     PS *pPS = new PS(pDM);

     PM *pPM = new PM(pPS);

     HolidayRequest *pHolidayRequest = new HolidayRequest(10);

     pPM->HandleRequest(pHolidayRequest);

     SAFE_DELETE(pHolidayRequest);

 

     pHolidayRequest = new HolidayRequest(2);

     pPM->HandleRequest(pHolidayRequest);

 

     SAFE_DELETE(pDM);

     SAFE_DELETE(pPS);

     SAFE_DELETE(pPM);

     SAFE_DELETE(pHolidayRequest);

}

优缺点

1.降低耦合度;职责链模式使得一个对象不用知道是哪一个对象处理它的请求。对象仅需要知道该请求会被正确的处理。接收者和发送者都没有对方的明确的信息,且链中的对象不需要知道链的结构;

2.增强了给对象指派职责的灵活性;当在对象中分派职责时,职责链给你更多的灵活性。你可以通过在运行时对该链进行动态的增加或修改来增加或改变处理一个请求的那些职责;

3.不保证被接受,既然一个请求没有明确的接收者,那么就不能保证它一定会被处理;该请求可能一直到链的末端都得不到处理。一个请求也可能因该链没有被正确配置而得不到处理。

总结

职责链模式在实现时,需要处理好它的后继者的问题,就是说,如果我不处理这个请求,那么我将把这个请求发给谁去处理呢?同时,职责链模式在实现时,它的链的形状并不是由职责链本身建立和维护的,而是由客户进行创建的,由客户指定每一个处理者的后继者是谁。这就大大的提高了职责链的灵活性。在实际中,我们也可以将职责链模式与组合模式相结合,一个构件的父构件可以作为它的后继者。

 类似资料:
  • 介绍 职责链模式(Chain of responsibility)是使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系。将这个对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理他为止。 也就是说,请求以后,从第一个对象开始,链中收到请求的对象要么亲自处理它,要么转发给链中的下一个候选者。提交请求的对象并不明确知道哪一个对象将会处理它——也就是该请求有一个隐式的接受者(i

  • 本文向大家介绍JavaScript设计模式之职责链模式应用示例,包括了JavaScript设计模式之职责链模式应用示例的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了JavaScript设计模式之职责链模式。分享给大家供大家参考,具体如下: 一、职责链的定义: 使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系,将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对

  • 责任链(Chain Of Responsibility) Intent 使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这些对象连成一条链,并沿着这条链发送该请求,直到有一个对象处理它为止。 Class Diagram Handler:定义处理请求的接口,并且实现后继链(successor) Implementation // java public abstract c

  • 本文向大家介绍学习JavaScript设计模式之责任链模式,包括了学习JavaScript设计模式之责任链模式的使用技巧和注意事项,需要的朋友参考一下 一、定义 责任链模式:使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系,将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。 二、示例 假设这么一个场景: 我们负责一个售卖手机的电商网站,经过分别缴纳500

  • 本文向大家介绍Java设计模式之责任链模式简介,包括了Java设计模式之责任链模式简介的使用技巧和注意事项,需要的朋友参考一下 对于使用过宏的朋友应该知道,利用宏可以实现一个键绑定多个技能。例如如果排在前面的技能有CD,则跳过此技能,执行之后的技能。记得曾经玩DK,打怪的时候,就是用一个键,一直按就行了。在servlet里的doGet和doPost方法,我们一般都把doGet请求发动到doPost

  • 本文向大家介绍JAVA设计模式之责任链模式详解,包括了JAVA设计模式之责任链模式详解的使用技巧和注意事项,需要的朋友参考一下 在阎宏博士的《JAVA与模式》一书中开头是这样描述责任链(Chain of Responsibility)模式的:   责任链模式是一种对象的行为模式。在责任链模式里,很多对象由每一个对象对其下家的引用而连接起来形成一条链。请求在这个链上传递,直到链上的某一个对象决定处理