设计模式之结构型模式---装饰器模式
目录
- 1.概述
- 2.类图
- 3.应用场景及优缺点
- 3.1 应用场景
- 3.2 优缺点
- 3.2.1 优点
- 3.2.2 缺点
- 4.实现
- 4.1 案例类图
- 4.2 代码实现
- 4.2.1 定义抽象构建角色
- 4.2.2 定义具体构建角色
- 4.2.3 定义抽象装饰器角色
- 4.2.4 定义具体装饰角色
- 4.2.5 装饰器模式的使用
1.概述
装饰器模式是指在不改变现有对象结构的情况下,动态的给对象增加一些职责。它是一种用于替代继承的技术,通过一种无需定义子类的方式给对象动态增加职责,使用对象之间的关联关系取代类之间的继承关系。装饰器模式通过引入一个装饰类,在装饰类中既可以调用待装饰的原有类的方法,还可以增加新的方法,达到扩展原有功能的目的。例如造一个交通工具,这种交通工具刚开始时只能是在陆地上跑,也就是咱们的汽车,但是随着技术的发展,我们可以给我们的汽车增加在水中移动的功能,或者是在天上飞的功能。这就是装饰器模式的场景之一。后面会用Java语言的代码来实现这个例子。
2.类图

装饰器模式的主要角色有如下几个:
抽象构建角色(Component): 定义一个抽象接口,规范准备接收附加责任的对象
具体构建角色(ConcreteComponent): 实现抽象构建角色定义的接口。通过装饰器角色为其添加职责
抽象装饰器角色(Decorator) 实现抽象构建中定义的接口,并包含具体构建角色的实例,可以通过其子类扩展具体构建角色实例的功能
具体装饰角色(ConcreteDecorator1、ConcreteDecorator2): 实现抽象装饰的相关方法,并给具体构建角色添加新的职责(功能)
3.应用场景及优缺点
3.1 应用场景
当不能采用继承的方式扩展功能或者是采用继承的方式扩展功能时不好维护的情况下就可以使用装饰器模式。比如一个功能的定义时就有基础版,加强版,VIP版本等基于基础功能扩展出新功能的情况下,我们就可以采用装饰器模式。因为装饰器模式完全遵守开闭原则,即对修改关闭,对扩展开放。
3.2 优缺点
3.2.1 优点
装饰器模式的优点有四个,如下:
- 对于扩展对象的功能,装饰器模式比继承更加灵活,因为装饰器模式不会导致类的个数急剧增加。
- 使用装饰器模式可以让我们通过一种动态的方式扩展一个对象的功能,比如我们可以通过配置文件在运行的时候选择不同的具体装饰类,从而实现不同的行为
- 装饰器可以对一个对象进行多次装饰
- 具体构建类与具体装饰类可以独立变化,用户可以更具需要增加的具体构建类和具体装饰类。符合开闭原则
3.2.2 缺点
装饰器模式的缺点主要有两个,如下:
- 使用装饰器模式进行系统设计时将会产生很多小对象,大量的小对象会占用更多的系统资源,在一定程度上会影响程序的性能
- 装饰器模式使用起来比继承更容易出错,排除错误也会更困难,特别是对于多次装饰的对象,调试时寻找错误可能需要逐级排查。
4.实现
我们以一个汽车改造成水陆两用汽车,陆空两用汽车的例子介绍装饰器模式在程序中的实现,我们用Java语言实现。
假设我们要造一个代步的交通工具,这个交通工具第一版本只需要在陆地上跑就行,也就是我们的汽车,然后随着技术增长,我们需要为我们的汽车扩展在水中行驶的能力和在空中飞行的能力,使用Java代码实现如下所示:
4.1 案例类图

4.2 代码实现
4.2.1 定义抽象构建角色
首先我们定义一个ITransportation接口,实现我们交通工具的基本能力—移动,这个接口就是我们装饰器模式中的抽线构建角色
public interface ITransportation {void move();
}
4.2.2 定义具体构建角色
定义好抽象构建角色后,我们就可以定义具体构建角色,也就是我们的汽车类了,在汽车类里实现了抽象构建角色定义的功能
public class Car implements ITransportation {public Car(){System.out.println("我只是单纯的一辆车");}@Overridepublic void move() {System.out.println("我可以在陆地上移动");}
}
4.2.3 定义抽象装饰器角色
接下来就需要定义抽象装饰器角色准备扩展功能,为了能扩展汽车的能力,我们提供了一个Transformer类,这个类也实现了抽象构建角色中的ITransportation接口,并且将抽象构建角色对应的具体构建角色通过构造函数注入到Transformer类中,这里的Transformer类就相当于抽象装饰器角色。我们需要的扩展功能后的对象都需要继承它。
public class Transformer implements ITransportation {private final ITransportation mTransportation;public Transformer(ITransportation transportation){this.mTransportation = transportation;}@Overridepublic void move() {mTransportation.move();}
}
4.2.4 定义具体装饰角色
具体装饰角色有两个,分别是水陆两用的汽车AmphibiousCar ,陆空两用的汽车FlyingCar,在这两个具体装饰角色中添加了各自的特色功能。
public class AmphibiousCar extends Transformer{public AmphibiousCar(ITransportation transportation) {super(transportation);System.out.println("我是水陆两用汽车");}public void moveInWater(){System.out.println("我可以在水里跑");}
}
public class FlyingCar extends Transformer{public FlyingCar(ITransportation transportation) {super(transportation);System.out.println("我是陆空两用汽车");}public void flying(){System.out.println("我可以在天上飞");}
}
4.2.5 装饰器模式的使用
使用时我们先通过抽象构建角色创建出具体构建角色的对象,这个对象时需要被装饰的对象,然后将其传递到陆空两用车类的构造函数中和水陆两用车类的构造函数中,分别创建出陆空两用车的对象,水陆两用车的对象。然后调用对应的方法实现各自的能力
public class Client {public static void main(String[] args) {ITransportation myCar = new Car();myCar.move();System.out.println("============================");System.out.println("===========陆空两用车=============");FlyingCar flyingCar = new FlyingCar(myCar);flyingCar.move();flyingCar.flying();System.out.println("=============水陆两用车======================");AmphibiousCar amphibiousCar = new AmphibiousCar(myCar);amphibiousCar.move();amphibiousCar.moveInWater();}
}
运行结果如下所示:

相关文章:
设计模式之结构型模式---装饰器模式
目录 1.概述2.类图3.应用场景及优缺点3.1 应用场景3.2 优缺点3.2.1 优点3.2.2 缺点 4.实现4.1 案例类图4.2 代码实现4.2.1 定义抽象构建角色4.2.2 定义具体构建角色4.2.3 定义抽象装饰器角色4.2.4 定义具体装饰角色4.2.5 装饰器模式的使用 1.概述 装饰器模式是指在不改变现有对…...
Android Pair
Pair在Android中是一种轻量级的工具类,并不是严格意义上的数据结构。 数据结构是一组有组织的方式来存储和管理数据的方式,如数组、链表、栈、队列、树、图等,它们有自己的特性和操作规则。而Pair更像是一个简单的封装,用于在需要…...
华为荣耀曲面屏手机下面空白部分设置颜色的方法
荣耀部分机型下面有一块空白区域,如下图红框部分 设置这部分的颜色需要在themes.xml里面设置navigationBarColor属性 <item name"android:navigationBarColor">android:color/white</item>...
《C#语法一篇通》,有20万字,需8MB字节,宜48小时阅读,没准会继续完善
本文摘录了C#语法的主要内容,接近20万字。 所有鸡汤的味道都等于马尿! 如果你相信任何所谓的鸡汤文章,智商堪忧。 计算机语言没有”好不好“之说,骗子才会告诉你哪个语言好,学好任何一本基础语言(C&#…...
嵌入式硬件工程师的职业发展规划
嵌入式硬件工程师可以按照以下阶段进行职业发展规划: 1. **初级阶段(1-3 年) ** - **技术学习与积累**: **电路基础强化**: 深入学习模拟电路和数字电路知识,能够熟练分析和设计基本的电路,…...
QT for android 问题总结(QT 5.15.2)
1.配置好的sdk,显示设置失败 Android SDK Command-line Tools run. Android Platform-Tools installed. Command-line Tools (latest) 版本过高导致报错 ,下载一个低版本的latest ,替换掉之前latest中的文件。即可,latest 路径如…...
PyTorch实战-手写数字识别-MLP模型
1 需求 包懂,40分钟掌握PyTorch深度学习框架,对应神经网络算法理论逐行讲解用PyTorch实现图像分类代码_哔哩哔哩_bilibili 10分钟入门神经网络 PyTorch 手写数字识别_哔哩哔哩_bilibili pytorch tutorial: PyTorch 手写数字识别 教程代码 从零设计并训…...
(附项目源码)Java开发语言,基于Java的高校实验室教学管理系统的设计与开发 50,计算机毕设程序开发+文案(LW+PPT)
摘 要 随着高校实验室教学与管理的复杂性增加,传统的手动管理系统已经无法满足日益增长的需求。实验室教学不仅涉及到学生的教学安排和管理,还需要对实验设备、实验材料、实验室资源等进行有效的调配和管理。而目前实验室教学管理的各项工作,…...
【日常问题排查小技巧-连载】
线上服务CPU飙高排查 先执行 top,找到CPU占用比较高的进程 id,(比如 21448) jstack 进程 id > show.txt(jstack 21448 > show.txt) 找到进程中CPU占用比较高的线程,线程 id 转换为 16 进…...
elastic search查找字段的方法
一,比如:elastic search 查找id为“ien9292voewew”的方法 此id为主键id,意思就是唯一id,在ES中是_id, 在 Elasticsearch 中,如果你想要查找特定 ID 的文档,可以使用 _get API。以下是如何通过 RESTful 请求或使用 Python 客户端来查找 ID 为 ien9292voewew 的文档的方…...
MATLAB下的四个模型的IMM例程(CV、CT左转、CT右转、CA四个模型),附下载链接
基于IMM算法的目标跟踪。利用卡尔曼滤波和多模型融合技术,能够在含噪声的环境中提高估计精度,带图像输出 文章目录 概述源代码运行结果代码结构与功能1. 初始化2. 仿真参数设置3. 模型参数设置4. 生成量测数据5. IMM算法初始化6. IMM迭代7. 绘图8. 辅助函…...
无人机之中继通信技术篇
一、定义与原理 无人机中继通信技术是指通过无人机搭载中继设备,将信号从一个地点传输到另一个地点,从而延长通信距离并保持较好的通信质量。其原理类似于传统的中继通信,即在两个终端站之间设置若干中继站,中继站将前站送来的信号…...
阳光保险隐忧浮现:业绩与股价双双而下,张维功能否力挽狂澜?
10月28日晚间,作为国内新生代险企,也是一家赴港上市的保险集团——阳光保险(HK:06963)一口气对外正式披露了三则财务报告,分别是集团旗下阳光人寿和阳光财险今年前三季度未经审议的财务数据,以及截至三季度…...
【OJ题解】在字符串中查找第一个不重复字符的索引
💵个人主页: 起名字真南 💵个人专栏:【数据结构初阶】 【C语言】 【C】 【OJ题解】 目录 1. 引言2. 题目分析示例: 3. 解题思路思路一:双重循环思路二:哈希表 4. C代码实现5. 代码详解6. 时间和空间复杂度分析7. 优化方…...
处理配对和拆分内容 |【python技能树知识点1~2 习题分析】
目录 一、编程语言简史(配对)题目要求:程序设计: 二、 编程语言发明家(拆分)题目要求程序实现while和for循环 python技能树知识点中的一些习题练习和分析。熟悉python编程模式和逻辑。 一、编程语言简史&am…...
HBuilderX自定义Vue3页面模版
HBuilderX自定义Vue3页面模版 首先在HBuilderX工具下的任意一个项目添加新建自定义页面模版 新建模版文件,并打开进行编辑 vue3-setup-js.vue文件里填写样式模版(根据自己的需要进行修改) <template><view class"">&…...
计算机网络——TCP中的流量控制和拥塞控制
TCP中的流量控制和拥塞控制 流量控制 什么是流量控制 如果发送者发送数据过快,接收者来不及接收,那么就会出现分组丢失,为了避免分组丢失,控制发送者的发送速度,使得接收者来得及接收,这就是流量控制。 …...
BFV/BGV全同态加密方案浅析
本文主要为翻译内容,原文地址:Introduction to the BFV encryption scheme、https://www.inferati.com/blog/fhe-schemes-bgv 之前的一篇博客我们翻译了CKKS全同态加密方案的内容,但该篇上下文中有一些知识要点,作者在BFV/BGV中已…...
Elasticsearch 实战应用详解!
Elasticsearch 实战应用详解 一、概述 Elasticsearch 是一个高度可扩展的开源全文搜索引擎,它能够处理大量数据并提供实时搜索和分析能力。基于 Lucene 构建,Elasticsearch 通过简单的 RESTful API 接口隐藏了 Lucene 的复杂性,使全文搜索变…...
最新最全面的JAVA面试题免费下载
面对求职市场的激烈竞争,掌握全面且深入的Java知识已成为每一位Java开发者必不可少的技能。《2023最新版Java面试八股文》是一份精心整理的面试准备资料,旨在帮助广大开发者系统复习,从容应对Java及相关技术栈的面试挑战。这份文档不仅汇聚了…...
(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)
题目:3442. 奇偶频次间的最大差值 I 思路 :哈希,时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况,哈希表这里用数组即可实现。 C版本: class Solution { public:int maxDifference(string s) {int a[26]…...
线程同步:确保多线程程序的安全与高效!
全文目录: 开篇语前序前言第一部分:线程同步的概念与问题1.1 线程同步的概念1.2 线程同步的问题1.3 线程同步的解决方案 第二部分:synchronized关键字的使用2.1 使用 synchronized修饰方法2.2 使用 synchronized修饰代码块 第三部分ÿ…...
可靠性+灵活性:电力载波技术在楼宇自控中的核心价值
可靠性灵活性:电力载波技术在楼宇自控中的核心价值 在智能楼宇的自动化控制中,电力载波技术(PLC)凭借其独特的优势,正成为构建高效、稳定、灵活系统的核心解决方案。它利用现有电力线路传输数据,无需额外布…...
java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别
UnsatisfiedLinkError 在对接硬件设备中,我们会遇到使用 java 调用 dll文件 的情况,此时大概率出现UnsatisfiedLinkError链接错误,原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用,结果 dll 未实现 JNI 协…...
Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)
目录 1.TCP的连接管理机制(1)三次握手①握手过程②对握手过程的理解 (2)四次挥手(3)握手和挥手的触发(4)状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...
页面渲染流程与性能优化
页面渲染流程与性能优化详解(完整版) 一、现代浏览器渲染流程(详细说明) 1. 构建DOM树 浏览器接收到HTML文档后,会逐步解析并构建DOM(Document Object Model)树。具体过程如下: (…...
【git】把本地更改提交远程新分支feature_g
创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...
css的定位(position)详解:相对定位 绝对定位 固定定位
在 CSS 中,元素的定位通过 position 属性控制,共有 5 种定位模式:static(静态定位)、relative(相对定位)、absolute(绝对定位)、fixed(固定定位)和…...
【JavaWeb】Docker项目部署
引言 之前学习了Linux操作系统的常见命令,在Linux上安装软件,以及如何在Linux上部署一个单体项目,大多数同学都会有相同的感受,那就是麻烦。 核心体现在三点: 命令太多了,记不住 软件安装包名字复杂&…...
Rapidio门铃消息FIFO溢出机制
关于RapidIO门铃消息FIFO的溢出机制及其与中断抖动的关系,以下是深入解析: 门铃FIFO溢出的本质 在RapidIO系统中,门铃消息FIFO是硬件控制器内部的缓冲区,用于临时存储接收到的门铃消息(Doorbell Message)。…...
