我的编程空间,编程开发者的网络收藏夹
学习永远不晚

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

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

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;
}

运行结果

总结

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

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

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

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

每天一个设计模式之装饰者模式

作者按:《每天一个设计模式》旨在初步领会设计模式的精髓,目前采用javascript和python两种语言实现。诚然,每种设计模式都有多种实现方式,但此小册只记录最直截了当的实现方式 :)原文地址是:《每天一个设计模式之装饰者模式》欢迎关注
2023-01-31

Java结构型设计模式之装饰模式详解

装饰模式(DecoratorPattern)允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有类的一个包装。这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,提供了额外的功能
2023-05-14

python设计模式之装饰器模式怎么实现

今天小编给大家分享一下python设计模式之装饰器模式怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。装饰器模式装饰器
2023-06-30

java 中设计模式(装饰设计模式)的实例详解

java 中设计模式(装饰设计模式)的实例详解应用场景:在不对原有对象类进行修改的基础上,给一个或多个已有的类对象提供增强额外的功能. 我觉得可以从字面理解,装饰,装饰房子。房子可以看成原有的类。等于你把一个已经建好的房子按照自己的想法再装
2023-05-31

java设计模式学习之装饰模式

装饰模式:动态的给一个对象添加一些额外的职责,就增加功能来说,装饰模式比生成子类更加灵活。优点:装饰类和被装饰类可以独立发展,不会相互耦合,装饰模式是继承的一个替代模式,装饰模式可以动态扩展一个实现类的功能。缺点:多层装饰比较复杂。实例:给
2023-05-31

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录