设计模式C++描述----10.装饰(Decorator)模式
一. 举例
我之前做过一个文件系统就叫 MyFileSys 吧,后来的话,客户想加入一些附加功能,比如压缩、加密、杀毒之类的操作,这些附加操作没有先后顺序,比如你可以先压缩再加密,也可以先杀毒再压缩,等等。
这些附加功能是可选的,有的客户要这些功能,有的不要,有的要其中的几种附加功能等等。怎么设计呢?
第一种方案:
直接修改这个独立的文件系统 MyFileSys,对于不同的客户实现不同的文件系统。

后来随着客户的增多,发现维护和修改的工作量越来越大。因为每增加一个客户就要重新生成一个类,然后把客户想要的附加功能加入,更加郁闷的是,只针对一个客户有时也是要修改很多次,客户今天要这些附加功能,明天又想加入另外一些功能,这样改来改去,维护工作量也是很大的。
第二种方案:
后来改用第二种方案,实现一个单独的附加功能类,保持原文件系统不变,这样在客户端就可以轻松的加入一些附加功能。

代码如下:
- //定义一个对象接口,可以给这些对象动态地添加职责
- class FileSys
- {
- public:
- virtual ~FileSys()
- {
- }
- virtual void Operation()
- {
- }
- protected:
- FileSys()
- {
- }
- };
- //定义一个具体的对象
- class MyFileSys:public FileSys
- {
- public:
- MyFileSys()
- {
- }
- ~MyFileSys()
- {
- }
- void Operation()
- {
- cout<<"MyFileSys operation..."<<endl;
- }
- };
- //装饰抽象类
- class Decorator:public FileSys
- {
- public:
- Decorator(FileSys* fileSys)
- {
- this->_fileSys = fileSys;
- }
- virtual ~Decorator()
- {
- delete _fileSys;
- }
- void Operation()
- {
- }
- protected:
- FileSys* _fileSys;
- };
- //压缩装饰类
- class ZipDecorator:public Decorator
- {
- public:
- ZipDecorator(FileSys* fileSys):Decorator(fileSys)
- {
- }
- ~ZipDecorator()
- {
- }
- void Operation()
- {
- _fileSys->Operation(); //首先运行以前的功能
- this->AddedZipBehavior(); //附加功能
- }
- void AddedZipBehavior()
- {
- cout<<"Added Zip Behavior...."<<endl;
- }
- };
- //杀毒装饰类
- class KillVirDecorator:public Decorator
- {
- public:
- KillVirDecorator(FileSys* fileSys):Decorator(fileSys)
- {
- }
- ~KillVirDecorator()
- {
- }
- void Operation()
- {
- _fileSys->Operation();
- this->AddedKillVirBehavior();
- }
- void AddedKillVirBehavior()
- {
- cout<<"Added Kill Virus Behavior...."<<endl;
- }
- };
- //加密装饰类
- class EncryptDecorator:public Decorator
- {
- public:
- EncryptDecorator(FileSys* fileSys):Decorator(fileSys)
- {
- }
- ~EncryptDecorator()
- {
- }
- void Operation()
- {
- _fileSys->Operation();
- this->AddedEncrypeBehavior();
- }
- void AddedEncrypeBehavior()
- {
- cout<<"Added Encrypt Behavior...."<<endl;
- }
- };
- //////////////////////////////////////////////////////////////////////////
- //测试
- int main()
- {
- FileSys* fileSys = new MyFileSys();
- Decorator* dec1 = new ZipDecorator(fileSys); //在原文件系统上,加入压缩功能
- Decorator* dec2 = new KillVirDecorator(dec1); //在之前的基础上,加入杀毒功能
- Decorator* dec3 = new EncryptDecorator(dec2); //再加入加密功能
- dec3->Operation();
- return 0;
- }
这样之后,如果要添加附加功能,实现起来就很方便了。这种模式就是装饰模式。
二. 装饰模式
装饰模式:动态地给一个对象添加一些额外的职责。就增加功能来说,装饰模式相比生成子类更为灵活。

说明:
Componet,主要是定义一个接口,通过这个接口可以给这些对象(ConcreteComponent)添加职责。
Dectorator,装饰类,通过外类(ConcreteDecorator)来扩展Component 类的功能,对于Component来说,是无需知道这个抽象类的存在的。
ConcreteDecorator,具体装饰类,添加具体的附加功能。
优点:
1. 装饰类是为已有功能动态地添加更多功能的一种方式。
2. 有效地把类的核心职责和装饰功能区分开,而且可以去除相关类中重复的装饰逻辑。
三. 问题讨论
从上图可以看到 Decorator 是继承于 Component
的,也就和 ConcreteComponent 成了兄弟了,但是 Decorator 的作用却是修饰 ConcreteComponent
的,这点好像是很怪怪的!!最说不通的是Decorator 与 Component 是没有is-a关系的!!
我个人觉得:
1. 这个继承关系,不应该是我们要重点关注的。这里使用继承主要是为了要重用 Operation() 这个接口,以达修饰的目的。
2. 重点是 Decorator 与 Component 这个组合关系。装饰类里有一个Component 指针,正是由于它的存在才能修饰到具体的 Component 对象。
设计模式C++描述----10.装饰(Decorator)模式的更多相关文章
- 《Head First 设计模式》ch.3 装饰(Decorator)模式
设计原则 类应该对修改关闭,对扩展开放(开放-关闭原则).在每个地方使用开放-关闭原则是一种浪费,也没有必要,因为这通常会引入新的抽象层次,增加代码复杂度.需要把注意力集中在设计中最有可能改变的地方. ...
- 设计模式(八)装饰器模式Decorator(结构型)
设计模式(八)装饰器模式Decorator(结构型) 1. 概述 若你从事过面向对象开发,实现给一个类或对象增加行为,使用继承机制,这是所有面向对象语言的一个基本特性.如果已经存在的一个类缺少某些方法 ...
- php设计模式课程---7、装饰器模式如何使用
php设计模式课程---7.装饰器模式如何使用 一.总结 一句话总结: 装饰器的核心是获取了文章类整个类,而不是获取了文章内容,有了这个文章类,我想给你加多少装饰就给你加多少装饰(将文章这个类封装进去 ...
- 设计模式(九)——装饰者模式(io源码分析)
1 星巴克咖啡订单项目(咖啡馆): 1) 咖啡种类/单品咖啡:Espresso(意大利浓咖啡).ShortBlack.LongBlack(美式咖啡).Decaf(无因咖啡) 2) 调料:Milk.So ...
- 大型Java进阶专题(八)设计模式之适配器模式、装饰者模式和观察者模式
前言 今天开始我们专题的第八课了.本章节将介绍:三个设计模式,适配器模式.装饰者模式和观察者模式.通过学习适配器模式,可以优雅的解决代码功能的兼容问题.另外有重构需求的人群一定需要掌握装饰者模式. ...
- 装饰(Decorator)模式
1.装饰(Decorator)模式 动态给一个对象添加一些额外的职责.就增加功能来说,装饰模式比生成子类更为灵活.Component是定义一个对象接口.可以给这些对象动态地添加职责.Concre ...
- 设计模式之第10章-桥接模式(Java实现)
设计模式之第10章-桥接模式(Java实现) “一入软件深似海,从此早睡是路人.黑夜给了我黑色的眼睛,我却用他去寻找八阿哥.”“怎么了,又来那么多的感慨啊.”“还能有什么啊,老板是说让换个APP做,这 ...
- ④ 设计模式的艺术-10.装饰(Decorator)模式
职责 装饰模式是在不必改变原类文件和使用继承的情况下,动态的扩展一个对象的功能.它是通过创建一个包装对象,也就是装饰来包裹真实的对象. 装饰模式是一种用于代替继承的技术,无需通过继承增加子类就能扩展对 ...
- 【设计模式 - 9】之装饰者模式(Decorator)
1 模式简介 装饰者模式允许向一个现有的对象添加新的功能,同时又不改变其结构. 装饰者模式的思路是用"调料"对象将原始对象进行层层包裹,同时其属性.动作层层传递,达到最终 ...
随机推荐
- CVPR 2019 论文解读 | 小样本域适应的目标检测
引文 最近笔者也在寻找目标检测的其他方向,一般可以继续挖掘的方向是从目标检测的数据入手,困难样本的目标检测,如检测物体被遮挡,极小人脸检测,亦或者数据样本不足的算法.这里笔者介绍一篇小样本(few ...
- 冒泡排序--JavaScript描述
相信凡是编程入门的都接触过冒泡排序算法,排序算法在编程中经常用到. 1. code /** * 冒泡排序 * 1.比较的轮数等于总数 - 1 * 2.比较次数等于要比较的个数 - 1 * --比较从第 ...
- SpringBootSecurity学习(14)前后端分离版之 OAuth2.0介绍
登录总结 前面基本介绍了security的常规用法,同时介绍了JWT和它的一个简单实现,基本上开发中遇到的登录问题都能解决了,即使在分布式开发,或者微服务开发中实现登录也基本没有问题了.securit ...
- 使用JavaScript·求数组的最大值和最小值
前言 在数组中并没有提供arr.max()和arr.min()这样的方法.那么是不是可以通过别的方式实现类似这样的方法呢?那么今天我们就来整理取出数组中最大值和最小值的一些方法. 法一:其实利用 ...
- python正则表达式贪婪算法与非贪婪算法与正则表达式子模式的简单应用
先引入一下百度百科对于正则表达式的概念: 正则表达式是对字符串操作的一种逻辑公式,就是用事先定义好的一些特定字符.及这些特定字符的组合,组成一个“规则字符串”,这个“规则字符串”用来表达对字符串的一种 ...
- 03-Django基础概念和MVT架构
一.Django基础 掌握Django的 MVT 架构的使用 掌握Git管理源代码 主要内容 了解Django的 MVT 架构的使用流程 使用Django完成案例 : 书籍信息管理 MVC介绍 MVC ...
- jQuery插件编写学习中遇见的问题--attr prop
个人博客: https://chenjiahao.xyz 最近在学习jQuery的插件的编写,有两种方式,$.fn.extend以及$.extend,一种是作用于对象原型上,一种是直接作用于jQuer ...
- Python 3.8.0 正式版发布,新特性初体验
北京时间 10 月 15 日,Python 官方发布了 3.8.0 正式版,该版本较 3.7 版本再次带来了多个非常实用的新特性. 赋值表达式 PEP 572: Assignment Expressi ...
- Ubuntu16.04安装java6(jdk 1.6)
目录 下载安装包 安装 移动到指定位置并设置版本 设置环境变量 切换java版本 下载安装包 先到官网下载安装包. 安装 输入命令 chmod 777 jdk-6u45-linux-x64.bin s ...
- 导图梳理springboot手动、自动装配,让springboot不再难懂
什么是springboot 在学springboot之前,你必须有spring.spring mvc基础,springboot的诞生其实就是用来简化新Spring应用的初始搭建以及开发过程,该框架使用 ...