【设计模式】之装饰器模式

系列文章目录

  • 【设计模式】之模板方法模式

  • 【设计模式】之责任链模式

  • 【设计模式】之策略模式

  • 【设计模式】之工厂模式(三种)


前言

今天给大家介绍23种设计模式中的装饰器模式。🌈

一、什么是装饰器模式

装饰器模式(Decorator Pattern)是一种结构型设计模式,它允许你动态地给一个对象添加一些额外的职责。就增加功能来说,装饰器模式相比生成子类更为灵活。

在装饰器模式中,有一个抽象组件接口,所有具体组件和装饰器都实现了这个接口。装饰器持有一个指向抽象组件的引用,并通过递归的方式调用接口中的操作。每个装饰器都可以添加自己的功能,同时调用所装饰的对象的操作。

二、装饰器模式的角色

Component(抽象组件)定义了一个对象的接口,可以给这些对象动态地添加职责(即方法)
ConcreteComponent(具体组件)实现了Component接口,是装饰器要装饰的真实对象
Decorator(装饰器)
  • 持有一个指向Component对象的引用,并有一个和Component接口一致的接口。
  • 它可以给组件添加额外的职责(方法)。
  • 通常包含对Component对象的引用以及实现Component接口的方法,这些方法会调用Component接口中定义的操作,并可能在其前后添加新的行为
ConcreteDecorator(具体装饰器)实现了Decorator接口,是装饰器接口的具体实现类

三、示例

定义一个抽象组件:

public interface Person {
    Double cost();
    void show();
}

具体组件:

public class XiaoJie implements Person{
    @Override
    public Double cost() {
        return 0.0;
    }
    @Override
    public void show() {
        System.out.println("没穿衣服的小杰。");
    }
}

 定义装饰器

public abstract class ClothesDecorator implements Person{
    protected Person person;

    public ClothesDecorator(Person person) {
        this.person = person;
    }
}

 具体装饰器:

public class Shirt extends ClothesDecorator{
    public Shirt(Person person) {
        super(person);
    }

    @Override
    public Double cost() {
        return this.person.cost()+500;
    }

    @Override
    public void show() {
        this.person.show();
        System.out.println("买了一个体恤,累计消费:" + this.cost() + "元");
    }
}

public class Jeans extends ClothesDecorator{
    public Jeans(Person person) {
        super(person);
    }

    @Override
    public Double cost() {
        return this.person.cost()+200.0;
    }

    @Override
    public void show() {
        this.person.show();
        System.out.println("买了一条牛仔裤,累计消费:"+this.cost()+"元");
    }
}

public class Shoes extends ClothesDecorator{
    public Shoes(Person person) {
        super(person);
    }

    @Override
    public Double cost() {
        return this.person.cost()+1000.0;
    }

    @Override
    public void show() {
        this.person.show();
        System.out.println("买了一双鞋,一共消费:"+this.cost()+"元");
    }
}

测试:

public class test {
    public static void main(String[] args) {
        Person xiaoJie = new XiaoJie();
        xiaoJie = new Shirt(xiaoJie);
        xiaoJie = new Jeans(xiaoJie);
        xiaoJie = new Shoes(xiaoJie);
        xiaoJie.show();
        System.out.println("本次一共消费:"+xiaoJie.cost()+"元");
    }
}
/*
测试结果:

没穿衣服的小杰。
买了一个体恤,累计消费:500.0元
买了一条牛仔裤,累计消费:700.0元
买了一双鞋,一共消费:1700.0元
本次一共消费:1700.0元

*/

 四、应用场景

  1. 扩展类的功能:当需要给一个已经存在的类添加新的功能,但又不想通过继承来生成子类时,可以使用装饰器模式。这是因为继承会增加类的层次结构,可能导致类的数量爆炸式增长,而装饰器模式可以在不改变原有类结构的情况下,动态地给对象添加新的功能。
  2. 动态添加和撤销功能:装饰器模式允许在运行时动态地给对象添加新的功能,并且这些功能也可以动态地被撤销。这对于那些需要经常变化或需要灵活配置的功能来说非常有用。
  3. 为一组相似的类添加功能:如果有一组相似的类,它们都需要添加相同的功能,但是又不希望修改这些类的源代码,那么可以使用装饰器模式。通过为这些类创建一个统一的接口或抽象类,并创建一个装饰器类来包装这些类的对象,就可以在保持原有类结构不变的情况下,为这些类添加新的功能。
  4. 处理透明性和递归组合:装饰器模式可以透明地添加或撤销功能,这意味着用户在使用被装饰的对象时,不需要知道对象是否被装饰过。此外,装饰器模式还可以实现递归组合,即一个装饰器可以包含另一个装饰器,从而创建出更复杂的功能组合。

在实际应用中,装饰器模式可以用于许多场景,例如:

  • 在图形界面库中,可以使用装饰器模式来动态地改变控件的外观或行为。
  • 在网络编程中,可以使用装饰器模式来添加日志记录、性能监控等功能到现有的网络请求或响应对象中。
  • 在游戏开发中,可以使用装饰器模式来扩展游戏角色的能力或属性。
  • 在Web应用中,可以使用装饰器模式来动态地添加或撤销用户的权限或角色。

总之,装饰器模式是一种非常灵活和强大的设计模式,它可以在不改变现有类结构的情况下,动态地给对象添加新的功能或职责。

五、总结

优点

  • 动态扩展:装饰器模式允许在运行时动态地给一个对象添加新的功能或职责,而无需修改其原有结构。这使得代码更加灵活和可扩展。

  • 高内聚低耦合:通过组合而非继承来扩展对象的功能,有助于保持类的职责单一,实现高内聚。同时,由于装饰器与被装饰对象之间通过接口或抽象类进行交互,降低了它们之间的耦合度。

  • 透明性:对于使用装饰器模式的客户端代码来说,装饰过的对象与未装饰的对象在接口上是一致的,因此可以透明地使用装饰过的对象。客户端无需知道对象是否被装饰过,也无需关心装饰的具体细节。

  • 灵活性:装饰器模式允许在运行时通过组合不同的装饰器来创建具有不同功能组合的对象。这使得可以根据需要灵活地定制对象的行为。

缺点

  • 可能产生较多的对象:由于装饰器模式是通过组合多个装饰器来扩展对象的功能的,因此在使用时可能会产生较多的对象。这可能会导致内存占用增加和性能下降的问题。

  • 对装饰器的要求:装饰器需要与被装饰对象具有相同的接口或抽象类。如果接口或抽象类发生变化,可能需要修改所有的装饰器。这可能会增加维护成本。

  • 可能导致设计过度复杂化:如果过度使用装饰器模式,可能会导致设计过度复杂化。过多的装饰器类和接口可能会使代码难以理解和维护。

  • 递归调用:在某些情况下,装饰器可能会递归地调用自身或其他装饰器。这可能会导致无限递归或栈溢出的问题,需要特别注意。


总结

今天的分享就到这里,我们下期再见✋

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/592743.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

springmvc下

第二类初始化操作 multipartResolver应用 localeResolver应用 themeResolver应用 handlerMapping应用 handlerAdapter应用 handlerExceptionReslver requestToViewNameTranslator应用 viewResolver应用 flashMapManager应用 dispatcherServlet逻辑处理 processRequest处理web请…

【Flask 系统教程 5】视图进阶

类视图 在 Flask 中,除了使用函数视图外,你还可以使用类视图来处理请求。类视图提供了一种更为结构化和面向对象的方式来编写视图函数,使得代码组织更清晰,并且提供了更多的灵活性和可扩展性。 创建类视图 要创建一个类视图&am…

Docker高频使用命令

一、Docker常用命令总结 1.镜像命令管理 指令描述ls列出镜像build构建镜像来自Dockerfilehoistory查看历史镜像inspect显示一个或多个镜像的详细信息pull从镜像仓库拉取镜像push推送一个镜像仓库rm移除一个或多个镜像prune一处未使用的镜像,没有被标记或被任何容器…

初始化Linux或者Mac下Docker运行环境

文章目录 1 Mac下安装Docker2 Linux下安装Docker2.1 确定Linux版本2.2 安装Docker2.3 配置加速镜像 3 Docker安装校验4 安装docker-compose4.1 直接下载二进制文件4.2 移动二进制文件到系统路径4.3 设置可执行权限4.4 验证安装 1 Mac下安装Docker mac 安装 docker 还是比较方便…

哥白尼高程Copernicus DEM下载(CSDN_20240505)

哥白尼数字高程模型(Copernicus DEM, COP-DEM)由欧洲航天局(European Space Agency, 简称ESA或欧空局)发布,全球范围免费提供30米和90米分辨率DEM。COP-DEM是数字表面模型(DSM),它表示地球表面(包括建筑物、基础设施和植被)的高程。COP-DEM是经过编辑的D…

c++set和map

目录 一、set的使用 1、set对象的创建 2、multiset 二、map的使用 1、map对象的创建 2、map的operator[] 序列式容器:vector、list、deque....单纯的存储数据,数据和数据之间没有关联 关联式容器:map、set.....不仅仅是存储数据&#x…

2000-2020年县域创业活跃度数据

2000-2020年县域创业活跃度数据 1、时间:2000-2020年 2、指标:地区名称、年份、行政区划代码、经度、纬度、所属城市、所属省份、年末总人口万人、户籍人口数万人、当年企业注册数目、县域创业活跃度1、县域创业活跃度2、县域创业活跃3 3、来源&#…

python数据可视化:显示两个变量间的关系散点图scatterplot()

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 python数据可视化: 显示两个变量间的关系 散点图 scatterplot() [太阳]选择题 请问关于以下代码表述错误的选项是? import seaborn as sns import matplotlib.pyplot …

VISO流程图之子流程的使用

子流程的作用 整个流程图的框图多而且大,进行分块;让流程图简洁对于重复使用的流程,可以归结为一个子流程图,方便使用,避免大量的重复性工作; 新建子流程 方法1: 随便布局 框选3 和4 &#…

SQL:NOT IN与NOT EXISTS不等价

在对SQL语句进行性能优化时,经常用到一个技巧是将IN改写成EXISTS,这是等价改写,并没有什么问题。问题在于,将NOT IN改写成NOT EXISTS时,结果未必一样。 目录 一、举例验证二、三值逻辑简述三、附录:用到的S…

3.3Java全栈开发前端+后端(全栈工程师进阶之路)-前端框架VUE3框架-企业级应用-Vue组合式API

为什么要使用Composition API 一个Options API实例 在前面的课程中&#xff0c;我们都是采用 Options API&#xff08;基于选项的 API &#xff09; 来写一个组件的。下面是一个实例&#xff1a; <template> Count is: {{ count }}, doubleCount is: {{ doubleCount…

深入理解网络原理3----TCP核心特性介绍(上)【面试高频考点】

文章目录 前言TCP协议段格式一、确认应答【保证可靠性传输的机制】二、超时重传【保证可靠性传输的机制】三、连接管理机制【保证可靠性传输的机制】3.1建立连接&#xff08;TCP三次握手&#xff09;---经典面试题3.2断开连接&#xff08;四次挥手&#xff09;3.3TCP状态转换 四…

【skill】onedrive的烦人问题

Onedrive的迷惑行为 安装Onedrive&#xff0c;如果勾选了同步&#xff0c;会默认把当前用户的数个文件夹&#xff08;桌面、文档、图片、下载 等等&#xff09;移动到安装时提示的那个文件夹 查看其中的一个文件的路径&#xff1a; 这样一整&#xff0c;原来的文件收到严重影…

政安晨:【Keras机器学习示例演绎】(三十五)—— 使用 LayerScale 的类注意图像变换器

目录 简介 导入 层刻度层 随机深度层 类注意力 会说话的头注意力 前馈网络 其他模块 拼凑碎片&#xff1a;CaiT 模型 定义模型配置 模型实例化 加载预训练模型 推理工具 加载图像 获取预测 关注层可视化 结论 政安晨的个人主页&#xff1a;政安晨 欢迎 &#…

Topaz Video AI 5.0.3激活版 AI视频无损缩放增强

Topaz Video AI专注于很好地完成一些视频增强任务&#xff1a;去隔行&#xff0c;放大和运动插值。我们花了五年时间制作足够强大的人工智能模型&#xff0c;以便在真实世界的镜头上获得自然的结果。 Topaz Video AI 还将充分利用您的现代工作站&#xff0c;因为我们直接与硬件…

【数学建模】矩阵微分方程

一、说明 我相信你们中的许多人都熟悉微分方程&#xff0c;或者至少知道它们。微分方程是数学中最重要的概念之一&#xff0c;也许最著名的微分方程是布莱克-斯科尔斯方程&#xff0c;它控制着任何股票价格。 ​​ 股票价格的布莱克-斯科尔斯模型 微分方程可以由数学中的许多…

MidJourney提示词大全

大家好&#xff0c;我是无界生长。 这篇文章分享一下MidJourney提示词&#xff0c;篇幅内容有限&#xff0c;关注公众号&#xff1a;无界生长&#xff0c;后台回复&#xff1a;“MJ”&#xff0c;获取全部内容。 我是无界生长&#xff0c;如果你觉得我分享的内容对你有帮助&…

ArcGIS软件:地图投影的认识、投影定制

这一篇博客介绍的主要是如何在ArcGIS软件中查看投影数据&#xff0c;如何定制投影。 1.查看地图坐标系、投影数据 首先我们打开COUNTIES.shp数据&#xff08;美国行政区划图&#xff09;&#xff0c;并点击鼠标右键&#xff0c;再点击数据框属性就可以得到以下的界面。 我们从…

【Mac】graphpad prism for Mac(专业医学绘图工具) v10.2.3安装教程

软件介绍 GraphPad Prism for Mac是一款专业的科学数据分析和绘图软件&#xff0c;广泛用于生物医学和科学研究领域。它具有强大的统计分析功能&#xff0c;可以进行各种数据分析&#xff0c;包括描述性统计、生存分析、回归分析、方差分析等。同时&#xff0c;它还提供了丰富…

C++奇迹之旅:string类接口详解(上)

文章目录 &#x1f4dd;为什么学习string类&#xff1f;&#x1f309; C语言中的字符串&#x1f309;string考察 &#x1f320;标准库中的string类&#x1f309;string类的常用接口说明&#x1f320;string类对象的常见构造 &#x1f6a9;总结 &#x1f4dd;为什么学习string类…
最新文章