zoukankan      html  css  js  c++  java
  • 【设计模式】命令模式

    1、 定义

    1.1    标准定义

      命令模式是一个高内聚的模式,其定义为:

      Encapsulate a request as an object,therebyletting you parameterize clients with different requests,queue or log requests,and support undoableoperations.(将一个请求封装成一个对象,从而让你使用不同的请求把客户端参数化,对请求排队或者记录请求日志,可以提供命令的撤销和恢复功能。)

    1.2 类图

      ● Receive接收者角色
      该角色就是干活的角色, 命令传递到这里是应该被执行的。

      ● Command命令角色
      需要执行的所有命令都在这里声明。

      ● Invoker调用者角色
      接收到命令, 并执行命令。

    2、实现

    2.1 类图

      Command类,用来声明执行操作的接口;

      ConcreteCommand,将一个接收者对象绑定于一个操作,调用接收者相应的操作,以实现Execute;

      Invoker类,要求该命令执行这个请求;

      Receiver类,知道如何实施与执行一个与请求相关的操作,任何类都可能作为一个接收者。

      Command模式通过将请求封装到一个对象Command中,并将请求的接收者存放到具体的ConcreteCommand类中,从而实现调用操作的对象和操作的具体实现者之间的解耦。

      Command模式结构图中,将请求的接收者(处理者)放到Command的具体子类ConcreteCommand中,当请求到来时(Invoker发出Invoke消息激活Command对象),ConcreteCommand将处理请求交给Receiver对象进行处理。

    2.2 代码

    2.2.1 command类

    // Command.h
    
    #ifndef _COMMAND_H_
    #define _COMMAND_H_
    
    class Command
    {
    public:
        virtual ~Command();
        virtual void Execute()=0;
    protected:
        Command();
    private:
    };
    
    class Receiver;
    
    class ConcreteCommand : public Command
    {
    public:
        ConcreteCommand(Receiver* pReceiver);
        ~ConcreteCommand();
        virtual void Execute();
    protected:
    private:
        Receiver* _recv;
    };
    
    class Invoker
    {
    public:
        Invoker(Command* pCommand);
        ~Invoker();
        void Invoke();
    protected:
    private:
        Command* _cmd;
    };
    
    class Receiver
    {
    public:
        Receiver();
        ~Receiver();
        void Action();
    protected:
    private:
    };
    
    #endif
    // Command.cpp
    
    #include "Command.h"
    #include <iostream>
    
    using namespace std;
    
    Command::Command(){}
    
    Command::~Command(){}
    
    ConcreteCommand::ConcreteCommand(Receiver* pReceiver)
    {
        this->_recv = pReceiver;
    }
    
    ConcreteCommand::~ConcreteCommand(){}
    
    void ConcreteCommand::Execute()
    {
        this->_recv->Action();
    }
    
    Receiver::Receiver(){}
    
    Receiver::~Receiver(){}
    
    void Receiver::Action()
    {
        cout << "Receiver::Action" << endl;
    }
    
    Invoker::Invoker(Command* pCommand)
    {
        this->_cmd = pCommand;
    }
    
    Invoker::~Invoker(){}
    
    void Invoker::Invoke()
    {
        this->_cmd->Execute();
    }

    2.2.2 调用

    // main.cpp
    
    #include "Command.h"
    
    int main()
    {
        //创建具体命令对象pCmd并设定它的接收者pRev
        Receiver* pRev = new Receiver();
        Command* pCmd = new ConcreteCommand(pRev);
        //请求绑定命令
        Invoker* pInv = new Invoker(pCmd);
        pInv->Invoke();
    
        return 0;
    }

    3、优缺点

    3.1 优点

      ● 类间解耦
      调用者角色与接收者角色之间没有任何依赖关系, 调用者实现功能时只需调用Command抽象类的execute方法就可以, 不需要了解到底是哪个接收者执行。

      ● 可扩展性
      Command的子类可以非常容易地扩展, 而调用者Invoker和高层次的模块Client不产生严重的代码耦合。

      ● 命令模式结合其他模式会更优秀
      命令模式可以结合责任链模式, 实现命令族解析任务; 结合模板方法模式, 则可以减少Command子类的膨胀问题。

      ● 可操作性
      允许接收请求的一方决定是否要否决请求;可以容易地实现对请求的撤销和重做。

    3.2 缺点

      命令模式也是有缺点的,请看Command的子类:如果有N个命令,问题就出来了,Command的子类就可不是几个,而是N个,这个类膨胀得非常大,这个就需要在项目中慎重考虑使用。

  • 相关阅读:
    安装acdsee 3.1后出错
    周末游山
    SQL Server不存在或访问被拒绝 Windows里的一个bug
    将数据库的存储过程及其参数导出来的方法
    VS出错问题集锦
    25 网页设计之页脚设计灵感
    需求沟通和实施
    Mootools插件闪烁的标题
    最近常常干出一些骑着驴找驴的事来
    php表单在提交之后再后退,表单的内容默认是被清空的
  • 原文地址:https://www.cnblogs.com/ChinaHook/p/7247717.html
Copyright © 2011-2022 走看看