C++实现设计模式之装饰者模式详解

 更新时间:2021年09月29日 10:15:46   作者:今天也要努力搬砖  
这篇文章主要介绍了C++设计模式之装饰模式,装饰模式能够实现动态的为对象添加功能,是从一个对象外部来给对象添加功能,需要的朋友可以参考下

设计模式和设计原则

装饰者模式动态地将责任附加到对象上。若要扩展功能,装饰者模式提供了比继承更有弹性的替代方案。

装饰者模式遵循的设计原则

类应该对扩展开放,对修改关闭。

装饰者模式中的类

装饰者模式中的类如下图,C++设计模式之装饰模式

其中Component抽象组件类,即被装饰的类,每个组件都可以单独使用,或者被装饰者包起来使用。该类中声明了一些接口,这些接口将在具体组件,以及具体装饰者中实现。

ConcreteComponent具体组件类,继承自组件类,是我们要动态地加上新行为(即要装饰)的对象。

Decorator抽象装饰者类是具体装饰者的基类,装饰者类中包含一个组件类型的指针,是为了记录被装饰的对象。当需要获得装饰之后的行为时可以通过该指针获得被装饰者的行为加上装饰者自身的行为,这个在下面案例中会看到。

ConcreteDecorator具体装饰者类,装饰者类要实现Decorator中定义的方法,另外可以加一些新的方法。

如上图所示装饰者和被装饰者必须是一样地类型,也就是有共同地超类,这是相当关键的地方。这是因为装饰者必须能取代被装饰者。

案列描述

咖啡馆提供各种各样的咖啡,每种咖啡可以加不同调料(摩卡、奶泡,双糖,半糖等)。以饮料为抽象组件,各种各样的咖啡为具体组件,咖啡中的不同调料为装饰者。每种咖啡以及调料有各自的描述和价格,使用装饰者模式,加了不同调料的咖啡也可以轻松给出描述和价格。

下面例子中生产三种咖啡:不加调料的Espresso、加双倍摩卡一份奶泡的DarkRoast和加双倍奶泡一份摩卡的HouseBlend.。

代码实现

声明:类的声明和实现在同一个文件里是个坏习惯,坏习惯,坏习惯,但因为我懒,还是写一起了,大家不要效仿,要引以为戒,要引以为戒,要引以为戒。

首先定义抽象组件类Beverage和抽象装饰者类CondimentDecorator,代码如下。这里关键的地方是抽象装饰者继承自抽象组件,且包含一个抽象组件的引用。

//抽象组件类-饮料
class Beverage
{
public:
	Beverage() :m_description("Unknown Beverage")
	{
	}
	virtual std::string getDescription(){ return m_description; }
	virtual double cost() = 0;
protected:
	std::string m_description;
};
//抽象装饰者类-调料,继承自饮料类
class CondimentDecorator :public Beverage
{
public:
	CondimentDecorator(Beverage* berverge) 
		:m_beverage(berverge)
	{
	}
	virtual std::string getDescription() = 0;//定义成纯虚函数,是为了强制子类实例化时必须实现它。
protected:
	Beverage* m_beverage;
};

然后定义具体组件,即三种具体的咖啡DarkRoast、Espresso和HouseBlend。

//三个具体组件
class DarkRoast :public Beverage
{
public:
	DarkRoast()
	{
		m_description = "DarkRoast";
	}
	double cost()
	{
		return 2.99;
	}
};
class Espresso :public Beverage
{
public:
	Espresso()
	{
		m_description = "Espresso";
	}
	double cost()
	{
		return 1.99;
	}
};
class HouseBlend :public Beverage
{
public:
	HouseBlend()
	{
		m_description = "HouseBlend";
	}
	double cost()
	{
		return 0.89;
	}
};

再定义两个具体装饰者类,即两种调料Mocha和Milk。

//两个具体装饰者
class Mocha :public CondimentDecorator
{
public:
	Mocha(Beverage* beverage) :CondimentDecorator(beverage)
	{
	}
	std::string getDescription()
	{
		return  m_beverage->getDescription() + " Mocha";
	}
	double cost()
	{
		return 0.2 + m_beverage->cost();
	}
};
class Milk :public CondimentDecorator
{
public:
	Milk(Beverage* beverage) :CondimentDecorator(beverage)
	{
	}
	std::string getDescription()
	{
		return  m_beverage->getDescription() + " Milk";
	}
	double cost()
	{
		return 0.5 + m_beverage->cost();
	}
};

最后在main函数中写测试代码,生产三种咖啡:不加调料的Espresso、加双倍摩卡一份奶泡的DarkRoast和加双倍奶泡一份摩卡的HouseBlend.

//测试代码
int main()
{
	//不加调料的Espresso
	Beverage* beverage = new Espresso();
	std::cout << beverage->getDescription() << " ¥" << beverage->cost() << std::endl;
	//加双倍摩卡和奶泡的DarkRoast
	Beverage* beverage2 = new DarkRoast();
	beverage2 = new Mocha(beverage2);
	beverage2 = new Mocha(beverage2);
	beverage2 = new Milk(beverage2);
	std::cout << beverage2->getDescription() << " ¥" << beverage2->cost() << std::endl;
	//加双倍奶泡的和一份摩卡的HouseBlend
	Beverage* beverage3 = new HouseBlend();
	beverage3 = new Mocha(beverage3);
	beverage3 = new Milk(beverage3);
	beverage3 = new Milk(beverage3);
	std::cout << beverage3->getDescription() << " ¥" << beverage3->cost() << std::endl;
	system("pause");
	delete beverage;
	delete beverage2;
	delete beverage3;
}

运行结果

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

相关文章

  • C语言switch语句详解

    C语言switch语句详解

    这篇文章主要为大家介绍了C语言switch语句,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2021-12-12
  • C++中md5 算法实现代码

    C++中md5 算法实现代码

    在网上找了份c++ MD5的代码,就简单保存一下,需要的朋友可以参考下
    2017-07-07
  • 谈谈vector的特殊性之为什么它不是STL容器

    谈谈vector的特殊性之为什么它不是STL容器

    这篇文章主要给大家介绍了关于vector的特殊性之为什么它不是STL容器的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用c++具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-08-08
  • 浅析C++浅拷贝与深拷贝的联系和区别

    浅析C++浅拷贝与深拷贝的联系和区别

    在c++中,深拷贝和浅拷贝也算是一个难点,特别是对于初学者来说,往往在不知道两者区别的情况下而错误的使用了浅拷贝,从而导致了野指针之类的问题,但是又因为缺少理解所以很难定位到问题所在
    2022-09-09
  • Qt使用事件与定时器实现字幕滚动效果

    Qt使用事件与定时器实现字幕滚动效果

    我们经常能够在外面看到那种滚动字幕,那么本文就拿Qt来做一个吧,本文将使用事件与定时器实现字幕滚动的效果,感兴趣的小伙伴可以了解一下
    2023-06-06
  • C语言编程C++自定义个性化类型

    C语言编程C++自定义个性化类型

    这篇文章主要介绍了C语言编程中如何来自定义C++个性化类型,文中附含详细的示例代码,有需要的朋友可以借鉴参考下,希望能够有所帮助
    2021-09-09
  • C++实现归并排序

    C++实现归并排序

    这篇文章主要为大家详细介绍了C++实现归并排序,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • C语言数据类型枚举enum全面详解示例教程

    C语言数据类型枚举enum全面详解示例教程

    生活中有很多地方会用到枚举,比如一周有7天,可以一一枚举;性别有男、女...等等都可以可以一一枚举,今天来和笔者一起学习一下c语言枚举吧
    2021-10-10
  • C/C++ Qt 数据库与Chart历史数据展示

    C/C++ Qt 数据库与Chart历史数据展示

    这篇文章主要介绍了Qt利用Qchart组件展示数据库中的历史数据。文中的示例代码讲解清晰,具有一定的学习和工作价值,感兴趣的小伙伴可以学习一下
    2021-12-12
  • C++常用的11种设计模式解释及示例代码详解

    C++常用的11种设计模式解释及示例代码详解

    c++常用的设计模式包括单例模式、工厂模式、抽象工厂模式、适配器模式、装饰者模式、代理模式、外观模式、桥接模式、组合模式、享元模式、观察者模式和命令模式等,这篇文章主要介绍了C++常用的11种设计模式解释及示例,需要的朋友可以参考下
    2023-02-02

最新评论