命令模式-C++实现

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

命令模式有四个角色:

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

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

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

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

举例:

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

c 复制代码
#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撤销命令。

测试:

c 复制代码
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();
}

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

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

输出结果:

c 复制代码
电灯已经打开
电灯已经关闭
电灯已经打开

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

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

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

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

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

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

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

优点:

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

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

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

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

缺点:

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

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

相关推荐
獨枭3 小时前
CMake 构建项目并整理头文件和库文件
c++·github·cmake
西猫雷婶3 小时前
python学opencv|读取图像(十九)使用cv2.rectangle()绘制矩形
开发语言·python·opencv
liuxin334455664 小时前
学籍管理系统:实现教育管理现代化
java·开发语言·前端·数据库·安全
码农W4 小时前
QT--静态插件、动态插件
开发语言·qt
ke_wu4 小时前
结构型设计模式
开发语言·设计模式·组合模式·简单工厂模式·工厂方法模式·抽象工厂模式·装饰器模式
code04号4 小时前
python脚本:批量提取excel数据
开发语言·python·excel
小王爱吃月亮糖4 小时前
C++的23种设计模式
开发语言·c++·qt·算法·设计模式·ecmascript
hakesashou5 小时前
python如何打乱list
开发语言·python
网络风云5 小时前
【魅力golang】之-反射
开发语言·后端·golang
Want5955 小时前
Java圣诞树
开发语言·python·信息可视化