设计模式-6--装饰者模式(Decorator Pattern)
一、什么是装饰者模式(Decorator Pattern)
装饰者模式(Decorator Pattern)是一种结构型设计模式,它允许你在不修改现有对象的情况下,动态地将新功能附加到对象上。这种模式通过创建一个包装类,即装饰者,来包含原始对象,并在其上添加额外的行为或功能。这样,你可以在运行时选择不同的装饰者组合来实现不同的功能组合。
装饰者模式的关键思想是将功能细分为一系列小的组件,然后将这些组件通过装饰者按照需要进行组合。这种模式遵循开放-关闭原则,即对扩展开放,对修改关闭。
主要角色:
- 组件(Component): 定义了一个抽象接口,可以是具体组件和装饰者共同实现的接口,表示被装饰者的基本功能。
- 具体组件(Concrete Component): 实现了组件接口的具体对象,即被装饰的原始对象。
- 装饰者(Decorator): 保持一个指向组件对象的引用,并实现与组件接口相同的接口。它可以有多个具体装饰者的子类。
- 具体装饰者(Concrete Decorator): 扩展装饰者的功能,包装具体组件,并可能添加新的行为。
装饰者模式的优势包括:
- 可以动态地组合对象,实现不同的功能组合,避免了类爆炸问题(大量子类的产生)。
- 遵循开放-关闭原则,允许在不修改现有代码的情况下扩展功能。
然而,装饰者模式也可能引入大量的小类,增加了代码的复杂性。在使用装饰者模式时,需要谨慎选择需要装饰的组件,以及如何合理地组合装饰者,以确保代码的可读性和维护性。
二、装饰者模式的代码样例
当用C++实现装饰者模式时,我们可以通过创建基类(组件)和派生类(具体组件、装饰者、具体装饰者)来演示。以下是一个简单的示例:
#include <iostream>// 组件基类
class Coffee {
public:virtual double cost() = 0;virtual ~Coffee() {}
};// 具体组件
class Espresso : public Coffee {
public:double cost() override {return 1.99;}
};// 装饰者基类
class Decorator : public Coffee {
protected:Coffee* coffee;
public:Decorator(Coffee* coffee) : coffee(coffee) {}
};// 具体装饰者
class Milk : public Decorator {
public:Milk(Coffee* coffee) : Decorator(coffee) {}double cost() override {return coffee->cost() + 0.5;}
};class Sugar : public Decorator {
public:Sugar(Coffee* coffee) : Decorator(coffee) {}double cost() override {return coffee->cost() + 0.2;}
};int main() {Coffee* espresso = new Espresso();std::cout << "Cost of espresso: $" << espresso->cost() << std::endl;Coffee* milkEspresso = new Milk(espresso);std::cout << "Cost of milk espresso: $" << milkEspresso->cost() << std::endl;Coffee* milkSugarEspresso = new Sugar(milkEspresso);std::cout << "Cost of milk and sugar espresso: $" << milkSugarEspresso->cost() << std::endl;delete espresso;delete milkEspresso;delete milkSugarEspresso;return 0;
}
在这个示例中,我们定义了 Coffee 基类和一个具体组件 Espresso。然后,我们定义了 Decorator 基类,它包含了一个指向 Coffee 对象的引用,并有两个具体装饰者类 Milk 和 Sugar,它们分别在 Coffee 上添加了牛奶和糖的装饰。
在 main 函数中,我们创建了一个 Espresso 对象,然后通过装饰者模式依次创建了包含不同装饰的咖啡对象,并输出了其价格。
这个示例展示了如何使用C++实现装饰者模式,动态地为对象添加功能。
三、使用装饰者模式需要注意的问题
在使用装饰者模式时,需要注意以下几个问题:
- 类爆炸: 装饰者模式可能会引入大量的小类,每个装饰者都是一个单独的类。这可能会导致类的数量急剧增加,增加代码复杂性和维护成本。因此,在选择使用装饰者模式时,需要仔细权衡增加的类数量是否值得所提供的灵活性和扩展性。
- 装饰者顺序: 装饰者模式中,装饰者的顺序可能会影响最终的对象组合。你需要确保装饰者的顺序不会引起意外的行为,特别是在组合多个装饰者时。
- 代码可读性: 过度使用装饰者模式可能会使代码变得难以理解和维护。因为每个具体装饰者只负责添加一小部分功能,当功能需要嵌套多层装饰者时,代码可能会变得冗长且难以阅读。
- 接口一致性: 在创建装饰者时,需要确保它们与组件(基类)具有一致的接口。这样,装饰者才能无缝地替代组件,而不会引发类型不匹配的问题。
- 不适合所有情况: 装饰者模式适用于需要动态地添加功能的情况。如果功能不太可能改变,或者只有固定数量的组合方式,那么使用装饰者模式可能会过于复杂,不切实际。
- 继承和组合的选择: 在设计时,需要权衡是否使用继承或组合。装饰者模式使用了组合,但过多的组合也可能使系统变得复杂。在一些情况下,简单的继承可能更合适。
- 性能影响: 使用装饰者模式可能会在运行时引入一些额外的开销,因为每个装饰者都会对对象进行包装和处理。这可能会在需要高性能的场景下造成问题。
总之,在使用装饰者模式时,需要根据实际情况谨慎权衡,考虑其带来的灵活性和复杂性,确保模式的应用不会导致代码难以维护或性能下降。
相关文章:

设计模式-6--装饰者模式(Decorator Pattern)
一、什么是装饰者模式(Decorator Pattern) 装饰者模式(Decorator Pattern)是一种结构型设计模式,它允许你在不修改现有对象的情况下,动态地将新功能附加到对象上。这种模式通过创建一个包装类,…...

质量属性案例-架构真题(二十一)
试题一 某电子商务公司升级会员与促销管理系统,向用户提交个性化服务,提高用户粘性。在项目建立之初,公司领导人一致认为目标是提升会员管理方式的灵活性,由于当前用户规模不大,用户简单,系统方面不需要做…...

nacos Error to process server push response
nacos2.0.3报错: Error to process server push response 解决办法: 排查项目当中有没有直接或间接依赖reflections: <dependency><groupId>org.reflections</groupId><artifactId>reflections</artifactId>…...

神经网络NLP基础 循环神经网络 LSTM
用的时候,只关心token的输入,以及hidden state就好了 sequence的length是多少,lstm的cell的数量就是多少 LSTM BI-LSTM stacked lstm GRU 实现...

Oracle数据传输加密方法
服务器端“dbhome_1\NETWORK\ADMIN\”sqlnet.ora文件中添加 SQLNET.ENCRYPTION_SERVER requested SQLNET.ENCRYPTION_TYPES_SERVER (RC4_256) 添加后新的链接即刻生效,服务器无需重新启动。 也可以通过Net manager管理工具添加 各个参数含义如下: 是…...

Android列表片段
下面创建第二个片段WorkoutFragment,它包含不同训练项目构成的一个列表,用户可以从这个列表中选择训练项目。 列表视图是只包含一个列表的片段 列表片段是一种专门处理列表的片段,它会自动绑定到一个列表视图,所以不需要另外创建…...

【元宇宙】智能手机万岁
凭借出色的新设备,我们很快就能进人元字宙,想象这样的情景是很趣的。但是,至少到21世纪20年代,元宇宙时代的大多数设备很可能是我们已经在使用的设备。 AR 和 VR 设备不仅面临重大的技术、财务和体验障碍,而且它们在上…...

华为mate60的发布代表着什么?有什么意义?
华为Mate60的发布代表着华为在技术领域的持续突破和创新。该产品预计将引入更强大的处理器、更高分辨率的屏幕、更强大的摄像头等前沿技术,进一步巩固华为在技术领域的领先地位。 此外,华为Mate60的发布还具有重塑品牌形象的意义。在美国制裁下ÿ…...

huggingface下载模型文件(基础入门版)
huggingface是一个网站,类似于github,上面拥有众多开源的模型、数据集等资料,人工智能爱好者可以很方便的上面获取需要的数据,也可以上传训练好的模型,制作的数据集等。本文只介绍下载模型的方法,适合新手入…...

在JS中tramsform与translate区别
在JavaScript中,"transform"和"translate"是用于处理HTML元素的样式属性,它们有以下区别: transform属性: transform属性是一个通用的属性,用于应用一系列的变换效果,包括平移、旋转、…...

ebay测评,物理环境与IP环境:解决平台风控问题的关键
近期eBay平台出现了大量风控问题,导致许多买家账号受到影响。实际上,这主要是由于环境搭建方面存在主要问题。时至2023年,许多人的技术方案仍停留在几年前,要么使用一键新机工具配合国外IP,要么使用指纹浏览器配合国外…...

05-Redis
1、Redis为什么快? 1、纯内存操作 2、单线程可以省去多线程时CPU上下文会切换的时间 3、渐进式ReHash、缓存时间戳 数组需要扩容的时候,他会维护两张hash表,比如第一张的数组长度为6,另一张的数组长度为12,在set和g…...

MSST-NET:用于高光谱和多光谱图像融合的多尺度空间-光谱Transfomer网络
1.网络结构 主要贡献: 提出了一种多尺度空间光谱Transformer网络光谱多头注意力旨在提取光谱特征引入多尺度波段/补丁嵌入来提取多尺度特征自监督训练 痛点:卷积核的感受野有限,基于卷积神经网络CNN的融合方法无法利用特征图中的全局关系…...

代码随想录笔记--二叉树篇
1--递归遍历 1-1--前序遍历 前序遍历:根→左→右; #include <iostream> #include <vector>struct TreeNode {int val;TreeNode *left;TreeNode *right;TreeNode() : val(0), left(nullptr), right(nullptr) {}TreeNode(int x) : val(x), le…...

JavaScript中包含对象的数组去重
一.数组遍历 function Uniarray(array) {// 接收去重后的数组let resArr [];// 遍历数组for (let i 0; i < array.length; i) {let isFind false;// 检查当前元素是否已存在于结果数组中for (let j 0; j < resArr.length; j) {// 使用严格相等运算符(&am…...

gRPC-GateWay Swagger 实战
上一次我们分享了关于 gRPC-Gateway 快速实战 ,可以查看地址来进行回顾 : 也可以查看关于 gRPC 的历史文章: gRPC介绍 gRPC 客户端调用服务端需要连接池吗? gRPC的拦截器 gRPC的认证 分享一下 gRPC- HTTP网关 I 今天主要是分享关于 gRPC-G…...

【webpack】HMR热更新原理
本文:参考文章 一、HMR是什么,为什么出现 1、出现的原因 之前,应用的加载、更新都是一个页面级别的操作,即使单个代码文件更新,整个页面都要刷新,才能拿到最新的代码同步到浏览器,导致会丢失…...

Ceph构件及组件分析
Ceph存储架构 Ceph 存储集群由几个不同的daemon组成,每个daemon负责Ceph 的一个独特功能并。每个守护进程是彼此独立的。 下面将简要介绍每个Ceph组件的功能: RADOS(Reliable Autonomic Distributed Object Store, RADOS) RADOS…...

第六章:中华民族的抗日战争
1.日本发动灭亡中国的侵略斗争 关键字: 中国抗日战争的起点与全民族抗战阶段 2.中国人民奋起抗击日本侵略者 关键字: 1 国共第二次统一战线初步建立的标志:国民党五届三中全会 2 扭转时局的枢纽,国内和平初步实现:…...

签到系统怎么设计
背景 相信签到系统大家都有接触过,更多的是使用。但是有思考过这种系统是怎么设计的吗?比方说我统计一下每个月中每天的签到情况,怎么设计呢?今天一篇文章告诉你。 首先,我们熟悉的思维是:我设计一个数据…...

危险的套娃:攻击者在 PDF 文件中隐藏恶意Word 文档
据BleepingComputer消息,日本计算机紧急响应小组 (JPCERT) 日前分享了在2023 年 7 月检测到的利用PDF文档的新型攻击——PDF MalDoc攻击,能将恶意 Word 文件嵌入 PDF 来绕过安全检测。 JPCERT采样了一种多格式文件,能被大多数扫描引擎和工具识…...

怎样将几个pdf合并?
在日常工作中,我们经常需要处理大量的PDF文件。有时候,我们需要将多个PDF文件合并成一个文件,以便于快速传输或方便查阅。虽然PDF文件本身不能进行编辑,但是借助专业的PDF编辑软件,我们可以轻松地实现将多个PDF文件合并…...

vr健康管理服务情景化教学弥补现代医学教学中的诸多不足之处
高职高专临床医学院校以培养岗位胜任力为目的,该专业是一门专业性、实践性较强的医学学科,要求培养出来的学生具有较强的临床实践能力,医学生所学的全部知识,都应与实践相结合,解决临床的实际问题,为患者解…...

【业务功能篇92】微服务-springcloud-多线程-异步处理-异步编排-CompletableFutrue
三、CompletableFutrue 一个商品详情页 展示SKU的基本信息 0.5s展示SKU的图片信息 0.6s展示SKU的销售信息 1sspu的销售属性 1s展示规格参数 1.5sspu详情信息 1s 1.ComplatableFuture介绍 Future是Java 5添加的类,用来描述一个异步计算的结果。你可以使用 isDone方…...

CAN FD的一致性测试 助力汽车电子智能化
后起之秀——CAN FD:随着各个行业的快速发展,消费者对汽车电子智能化的诉求越来越强烈,这使得整车厂将越来越多的电子控制系统加入到了汽车控制中,且在传统汽车、新能源汽车、ADAS和自动驾驶等汽车领域中也无不催生着更高的需求&a…...

微信短链跳转到小程序指定页面调试
首先说下背景:后端给了短链地址,但是无法跳转到指定页面。总是在小程序首页。指定的页面我们是h5页面。排查步骤如下: 1、通过快速URL Scheme 编译。上部普通编译 下拉找到此选项。 、 2、按照小程序的要求的URL Scheme输入。另外后端给的…...

机器学习——聚类算法一
机器学习——聚类算法一 文章目录 前言一、基于numpy实现聚类二、K-Means聚类2.1. 原理2.2. 代码实现2.3. 局限性 三、层次聚类3.1. 原理3.2. 代码实现 四、DBSCAN算法4.1. 原理4.2. 代码实现 五、区别与相同点1. 区别:2. 相同点: 总结 前言 在机器学习…...

【2023研电赛】安谋科技企业命题三等奖作品: 短临天气预报AI云图分析系统
本文为2023年第十八届中国研究生电子设计竞赛安谋科技企业命题三等奖分享,参加极术社区的【有奖活动】分享2023研电赛作品扩大影响力,更有丰富电子礼品等你来领!,分享2023研电赛作品扩大影响力,更有丰富电子礼品等你来…...

The Sandbox 与韩国仁川市合作,打造身临其境的城市体验内容
简要概括 ● The Sandbox 与仁川市联手展示城市魅力,打造创新形象。 ● 本次合作包含多种多样的活动,如 NFT 捐赠活动和针对元宇宙创作者的培训计划。 我们非常高兴地宣布与仁川市合作,共同打造身临其境的城市体验。 双方合作的目的是在国…...

JVM之堆和方法区
目录 1.堆 1.1 堆的结构 1.1.1 新生代(Young Generation) 1.1.2 年老代(Old Generation) 1.1.3 永久代/元空间(Permanent Generation/Metaspace) 1.2 堆的内存溢出 1.3 堆内存诊断 1.3.1 jmap 1.3.2…...