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项目
相关推荐
GISer_Jing8 小时前
智能体工具使用、规划模式
人工智能·设计模式·prompt·aigc
GISer_Jing8 小时前
AI Agent:学习与适应、模型上下文协议
人工智能·学习·设计模式·aigc
小马爱打代码10 小时前
MyBatis设计模式:构建者、工厂、代理模式
设计模式·mybatis·代理模式
月明长歌10 小时前
Javasynchronized 原理拆解:锁升级链路 + JVM 优化 + CAS 与 ABA 问题(完整整合版)
java·开发语言·jvm·安全·设计模式
会员果汁10 小时前
12.设计模式-状态模式
设计模式·状态模式
Yu_Lijing11 小时前
基于C++的《Head First设计模式》笔记——抽象工厂模式
c++·笔记·设计模式
会员果汁13 小时前
13.设计模式-适配器模式
设计模式·适配器模式
GISer_Jing1 天前
AI:多智能体协作与记忆管理
人工智能·设计模式·aigc
雨中飘荡的记忆1 天前
责任链模式实战应用:从理论到生产实践
设计模式
沛沛老爹1 天前
Web开发者进阶AI:Agent技能设计模式之迭代分析与上下文聚合实战
前端·人工智能·设计模式