当前位置: 首页 > article >正文

命令模式-C++实现

命令模式是一种行为型设计模式,它将请求封装成一个对象,从而能使你可以用不同的请求对客户端进行参数化。该模式允许请求的发送者和接收者进行解耦,发送者不需要知道接收者的信息,只需要通过命令对象来与它进行交互。

命令模式有四个角色:

1、抽象命令:它定义了执行操作的接口,包含一个执行方法和一个可选的撤销操作,这里的撤销是撤销命令,恢复成上一个命令执行的结果。

2、具体命令:实现了命令接口,持有接收者对象的引用,负责在接收者上执行操作。

3、接收者:执行命令所代表的操作

4、调用者:持有命令对象,发送请求并触发命令执行。

举例:

使用遥控器控制电灯的开和关。

#include <iostream>
#include <memory>

// 接收者-电灯
class Light
{
public:

	void On()
	{
		std::cout << "电灯已经打开" << std::endl;
	}

	void Off()
	{
		std::cout << "电灯已经关闭" << std::endl;
	}
};

// 抽象命令
class ICommand
{
public:

	virtual ~ICommand() {}

	virtual void Execute() = 0;

	virtual void Undo() = 0;

protected:

	std::shared_ptr<Light> light_;
};

// 具体命令-打开电灯
class CloseLight
	: public ICommand
{
public:

	CloseLight(std::shared_ptr<Light> _light)
	{
		light_ = _light;
	}

	virtual void Execute() override
	{
		light_->On();
	}

	virtual void Undo() override
	{
		light_->Off();
	}
};

// 具体命令-关闭电灯
class OpenLight
	: public ICommand
{
public:

	OpenLight(std::shared_ptr<Light> _light)
	{
		light_ = _light;
	}

	virtual void Execute() override
	{
		light_->Off();
	}

	virtual void Undo() override
	{
		light_->On();
	}
};

// 调用者-遥控器
class RemoteControl
{
public:

	void SetCommand(std::shared_ptr<ICommand> _command)
	{
		command_ = _command;
	}

	void PressButton()
	{
		if(command_)
			command_->Execute();
	}

	void Undo()
	{
		if(command_)
			command_->Undo();
	}

private:

	std::shared_ptr<ICommand> command_;
};

示例中,我们首先定义了一个抽象命令接口(ICommand),定义了两个方法Exectue()和Undo(),分别用于执行操作和撤销命令。

然后我们又创建了两个具体的命令类(OpenLight)和(CloseLight),分别实现了这两个方法。这些具体命令类会持有对接收者对象(Light)的引用,通过执行方法调用相应的操作。

最后创建了一个调用者角色(RemoteControl)作为遥控器。遥控器持有一个命令对象,提供设置命令对象和触发命令的执行方法。通过按下(PressButton)执行具体的命令,通过Undo撤销命令。

测试:

void TestCommand()
{
	// 创建接收者
	std::shared_ptr<Light> light = std::make_shared<Light>();

	// 创建命令
	std::shared_ptr<ICommand> openLight = std::make_shared<OpenLight>(light);
	std::shared_ptr<ICommand> closeLight = std::make_shared<CloseLight>(light);

	// 创建调用者
	std::shared_ptr<RemoteControl> remoteControl = std::make_shared<RemoteControl>();

	// 设置命令
	remoteControl->SetCommand(openLight);
	remoteControl->PressButton();
	remoteControl->Undo();

	remoteControl->SetCommand(closeLight);
	remoteControl->Undo();
}

测试代码中,我们创建了两个具体命令:打开电灯和关闭电灯、一个接收者,也就是电灯、一个遥控器对象。

通过遥控器设置命令,按下按钮,就可以执行具体的命令。

输出结果:

电灯已经打开
电灯已经关闭
电灯已经打开

可以看到,我们先设置命令为打开电灯,按下按钮,电灯已经打开,执行撤销方法,电灯就被关闭,然后我们设置命令为关闭电灯,执行撤销方法,电灯就被打开。

所以这里的撤销其实是撤销当前命令

命令模式遵顼的设计原则:

1、单一职责原则:每个命令类负责执行一个特定的命令。

2、开放封闭原则:可以动态的添加或删除命令,不影响现有代码。

3、里氏替换原则:命令模式中的具体命令类是抽象命令的子类,因此可以通过具体命令类的替换来扩展和改变命令的行为。

4、接口隔离原则:命令模式通过抽象命令和具体命令的设计,可以将不同的请求封装成不同的命令类,从而避免大量的接口在同一个类中定义。

优点:

1、解耦对象间的关系:命令模式将请求者和接收者解耦,使得命令发送者只需要知道抽象命令类,不需要知道具体的接收者,降低了系统的耦合度。

2、容易扩展新的命令:新增一个命令非常容易,不需要修改现有代码,符合开闭原则。

3、支持撤销和重做操作:命令模式可以将命令对象存储在历史记录中,实现命令的撤销和重做

4、支持队列请求和日志化请求:命令模式可以将命令对象放入队列中,实现对请求的排队和延迟执行,还可以将命令对象做持久化处理,实现对请求的日志记录。

缺点:

1、增加了系统的复杂度:引入了多个命令类、接收者类、调用类,增加了系统的复杂度。

2、可能会使类膨胀:每个命令都需要一个具体的命令类去实现,如果命令太多,就会造成类的数量过于膨胀,增加了系统的维护成本。


http://www.kler.cn/news/155720.html

相关文章:

  • 超硬核解析Mybatis动态代理原理!只有接口没实现也能跑?
  • Python WebSocket 客户端教程
  • maven如何用命令看配置文件位置
  • 如何绕过某讯手游保护系统并从内存中获取Unity3D引擎的Dll文件
  • Debian12配置ssh服务器
  • 用Java写一个王者荣耀游戏
  • Django rest froamwork-序列化关系
  • python 交互模式和命令行模式的问题
  • 【C++】类和对象——explicit关键字,友元和内部类
  • Linux(12):磁盘配额(Quota)与进阶文件系统管理
  • Linux系统中进程间通信(Inter-Process Communication, IPC)
  • 医院电子病历编辑器源码(支持云端SaaS服务)
  • 关于清空ant.design 中表单内容的方法
  • 分享77个焦点幻灯JS特效,总有一款适合您
  • 【llm使用】ChatGLM3-6B Transformers部署调用
  • 【力扣206】反转链表
  • Filebeat使用指南
  • RPA机器人如何确保敏感数据的安全性
  • 使用 OpenCV 识别和裁剪黑白图像上的白色矩形--含源码
  • 策略设计模式
  • Spring Boot 3.2 新特性之 RestClient
  • 【C语言】【数据结构】【手搓二叉树】用数组实现一个二叉树
  • Android 第三十九章 RatingBar
  • SpringMVC常用注解和用法总结
  • leetcode:468. 验证IP地址
  • viple模拟器使用(四):unity模拟器中实现两距离局部最优迷宫算法
  • javaee实验:文件上传及截器的使用
  • 迭代器模式-C++实现
  • Hive 安装部署
  • (11_29)畅捷通的 Serverless 探索实践之路