7. 命令模式

目录

一、解决的问题

  • 将行为请求者与行为实现者解耦,当用户发出命令后,无需关注谁来执行命令。
  • 将命令的发出者和命令的执行者完全隔离开。

二、设计类图

  • Client:创建具体的命令对象,并且设置命令对象的接收者
  • Invoker:命令传递者,要求命令对象执行请求
  • Command:定义命令的接口
  • ConcreteCommand:命令接口实现对象
  • Receiver:命令接收并执行

三、代码实现

3.1 设计类图

  • 以电视机遥控器作为背景进行,类图如下

3.2 代码实现

cpp 复制代码
#include <iostream>

class TV
{
private:
    int currentChannel = 0;

public:
    void changeChannel(int channel)
    {
        this->currentChannel = channel;
    }

    void turnOff()
    {
        std::cout << "TV is off." << std::endl;
    }

    void turnOn()
    {
        std::cout << "TV is on." << std::endl;
    }
};

class Command
{
public:
    virtual void execute() = 0;
};

class CommandOn : public Command
{
private:
    TV* myTV;
public:
    CommandOn(TV* tv)
    {
        myTV = tv;
    }

    void execute()
    {
        myTV->turnOn();
    }
};

class CommandOff : public Command
{
private:
    TV* myTV;
public:
    CommandOff(TV* tv)
    {
        myTV = tv;
    }

    void execute()
    {
        myTV->turnOff();
    }
};

class CommandChange : public Command
{
private:
    TV* myTV;
    int channel;
public:
    CommandChange(TV* tv, int channel)
    {
        myTV = tv;
        this->channel = channel;
    }

    void execute()
    {
        std::cout << "Switch Channel to " << channel << std::endl;
        myTV->changeChannel(channel);
    }
};

class Control
{
private:
    Command* changChannel;
    Command* offCommand;
    Command* onCommand;

public:
    Control(Command* changChannel, Command* off, Command* on)
    {
        this->changChannel = changChannel;
        this->offCommand = off;
        this->onCommand = on;
    }

    void changeChannel()
    {
        changChannel->execute();
    }
    
    void turnOff()
    {
        offCommand->execute();
    }

    void turnOn()
    {
        onCommand->execute();
    }
};

int main()
{
    TV* mytv = new TV();
    Command* on = new CommandOn(mytv);
    Command* off = new CommandOff(mytv);
    Command* channel = new CommandChange(mytv, 3);

    Control* control = new Control(channel, off, on);

    control->turnOn();
    control->changeChannel();
    control->turnOff();
    return 0;
}

四、扩展

  • 定义一个3对控制器的按钮
  • 分别控制客厅灯、厨房灯和车库门的开关操作
  • 类图如下
  • 运行结果
  • 代码部分请参阅 https://gitee.com/piglittle/design_patterns中的 Head_First_Design_Partterns解决方案下的 command_pattern项目
相关推荐
道199313 小时前
QT 工程中快速实现中英文切换(含动态切换)
命令模式
1024肥宅17 小时前
JavaScript常用设计模式完整指南
前端·javascript·设计模式
特立独行的猫a19 小时前
C++观察者模式设计及实现:玩转设计模式的发布-订阅机制
c++·观察者模式·设计模式
better_liang19 小时前
每日Java面试场景题知识点之-单例模式
java·单例模式·设计模式·面试·企业级开发
sg_knight20 小时前
什么是设计模式?为什么 Python 也需要设计模式
开发语言·python·设计模式
koping_wu20 小时前
【设计模式】设计模式原则、单例模式、工厂模式、模板模式、策略模式
单例模式·设计模式·策略模式
fpl111621 小时前
npm :无法加载文件 D:\...\nodejs\npm.ps1,因为在此系统上禁止运行脚本
前端·vscode·npm·node.js·命令模式
__万波__21 小时前
二十三种设计模式(九)--组合模式
java·设计模式·组合模式
__万波__21 小时前
二十三种设计模式(十)--外观模式
java·设计模式·外观模式
__万波__21 小时前
二十三种设计模式(十一)--享元模式
java·设计模式·享元模式