当前位置: 首页 > news >正文

设计模式之装饰器模式:让对象功能扩展更优雅的艺术

一、什么是装饰器模式

    装饰器模式(Decorator Pattern)是一种结构型设计模式(Structural Pattern),它允许用户通过一种灵活的方式来动态地给一个对象添加一些额外的职责。就增加功能来说,装饰器模式相比使用子类更为灵活。这种模式创建了一个包装对象,也就是装饰器,来包裹真实对象。

二、装饰器模式的原理

    装饰器模式的工作原理是通过创建一个包装对象,也就是装饰器,来包裹真实对象。装饰器通常会持有一个被装饰对象的引用,并在执行某些操作时,将这个请求委托给被装饰的对象,同时还可以在委托之前或之后添加一些附加操作。这样,我们就可以在不修改原有类结构的情况下,为对象添加新的功能或改变其行为。

三、装饰器模式的结构

    装饰器模式主要包含以下几个角色:

  1. 组件(Component):定义一个接口,装饰器和被装饰者共同需要实现的接口,且装饰器可以给其实现类动态地添加一些职责。

  2. 具体组件(Concrete Component):Component的实现类,定义了一个具体的对象,扮演被装饰的角色。

  3. 装饰器(Decorator):同样是Component的实现类,同时持有一个Component对象的引用,通过调用该引用的方法来实现相应接口。

  4. 具体装饰器(Concrete Decorator):Decorator的子类,负责向Component角色添加新的功能。

四、装饰器模式的应用场景

    装饰器模式非常适用于以下场景:

  1. 给对象添加额外的职责:当对象需要频繁地增加功能时,装饰器模式提供了一种灵活的方式来添加或移除这些功能,而无需修改对象的原有结构。

  2. 需要保持对象的接口不变:如果修改对象的接口会影响到许多其他的客户端代码,那么使用装饰器模式可以在不改变接口的情况下,为对象添加新的功能。

  3. 通过组合而非继承来扩展对象的功能:装饰器模式允许我们通过组合多个装饰器来扩展对象的功能,而不是通过创建大量的子类。这样可以避免类爆炸的问题,使系统更加灵活和可扩展。

  4. 运行时动态地添加或移除功能:在运行时,我们可以根据需要动态地添加或移除装饰器,从而改变对象的行为。这种灵活性是继承关系所不具备的。

五、装饰器模式的优缺点

5.1. 优点

  1. 可以在运行时动态地添加或移除功能,而无需修改现有的代码,更加灵活。

  2. 提供一种动态的方式来扩展一个对象的功能,在运行时选择不同的具体装饰器,从而实现不同的行为。

  3. 遵循开闭原则,对扩展开放,对修改关闭。这意味着我们可以在不修改现有代码的情况下,通过添加新的装饰器类来扩展系统的功能。

  4. 可以对一个对象进行多次装饰,通过使用不同的具体装饰类以及这些装饰类的排列组合可以创造出很多不同行为的组合,得到更加强大的对象.

5.2. 缺点

  1. 随着装饰器数量的增加,类的数量也会增加,这可能会使系统变得更加复杂和难以理解,且大量的对象还会占用系统更多资源,在一定程度上影响系统的性能。

  2. 由于装饰器模式本质上是一种递归调用,使得出现问题时排查错误较困难,需要逐级排查,较为繁琐。

六、装饰器模式示例

    假设我们有一个咖啡店,提供不同类型的咖啡和可以添加的配料(如牛奶、糖等),我们可以使用装饰器模式来实现这个需求。

6.1. 定义组件接口

    首先,我们定义一个咖啡的接口,它包含一个getDescription方法和一个cost方法来分别描述咖啡和计算价格:

public interface Beverage {String getDescription();double cost();
}

6.2. 创建具体组件

    然后,我们创建一个实现了Beverage接口的具体咖啡类,比如Espresso。

public class Espresso implements Beverage{@Overridepublic String getDescription() {return "Espresso";}
​@Overridepublic double cost() {return 1.99;}
}

6.3. 创建装饰器角色

    接下来,我们定义一个装饰器类,它实现了Beverage接口并持有一个Beverage对象的引用。

public abstract class CondimentDecorator implements Beverage{protected Beverage beverage;
​public CondimentDecorator(Beverage beverage) {this.beverage = beverage;}
​@Overridepublic String getDescription() {return beverage.getDescription();}
​@Overridepublic double cost() {return beverage.cost();}
}

6.4. 创建具体装饰角色

    最后,我们创建具体的装饰类,比如Milk和Sugar,它们都继承自CondimentDecorator并添加自己特有的功能(如描述和价格)。

public class Milk extends CondimentDecorator{public Milk(Beverage beverage) {super(beverage);}
​@Overridepublic String getDescription() {return beverage.getDescription() + ", Milk";}@Overridepublic double cost() {return .10 + beverage.cost();}
}
public class Sugar extends CondimentDecorator{public Sugar(Beverage beverage) {super(beverage);}
​@Overridepublic String getDescription() {return super.getDescription() + ", Sugar";}@Overridepublic double cost() {return .10 + super.cost();}
}

6.5. 创建客户端测试制作咖啡

    最后,我们通过一个客户端类来展示如何使用这些装饰器来构建和展示不同的咖啡组合。

public class MakeCoffee {public static void main(String[] args) {Beverage beverage = new Espresso();System.out.println(beverage.getDescription() + " $" + beverage.cost());
​Beverage beverage2 = new Milk(new Espresso());System.out.println(beverage2.getDescription() + " $" + beverage2.cost());
​Beverage beverage3 = new Milk(new Sugar(new Espresso()));System.out.println(beverage3.getDescription() + " $" + beverage3.cost());
​// 可以继续添加更多的装饰器,如摩卡、香草等}
}

6.6. 测试结果

    以下为示例运行结果:

    在上面的例子中,我们首先创建了一个Espresso对象,并直接打印了它的描述和价格。然后,我们创建了一个加牛奶的Espresso,即Milk(new Espresso()),并打印了它的描述和价格。最后,我们创建了一个既加牛奶又加糖的Espresso,即Milk(new Sugar(new Espresso())),并打印了它的描述和价格。这样,我们就可以看到装饰器模式如何动态地给对象添加额外的职责。

七、总结

    装饰器模式是一种强大的设计模式,它提供了一种灵活的方式来动态地给对象添加额外的职责。通过组合而非继承的方式,装饰器模式能够在不改变对象接口的情况下,为对象添加新的功能,从而提高了系统的灵活性和可扩展性。然而,我们也需要注意到装饰器模式的缺点,并在设计系统时权衡其利弊,以确保系统既灵活又易于理解和维护。

    在实际应用中,我们应该根据具体的需求和场景来选择合适的设计模式。对于需要动态扩展功能且保持接口不变的场景,装饰器模式无疑是一个值得考虑的选择。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • elementUI table 给表头添加气泡显示(鼠标悬浮显示注释)
  • Spring扩展点系列-InstantiationAwareBeanPostProcessor
  • 用ACF和PACF计算出一堆数据的周期个数以及周期时长,数据分析python
  • 【区块链 + 物联网】区块链边缘计算网关设备 | FISCO BCOS应用案例
  • 鸿蒙Next-拉起支付宝的三种方式——教程
  • 2024最新!Facebook手机版和网页版改名教程!
  • 【30天玩转python】条件语句与循环
  • RLC(电阻、电感、电容)
  • 简单的spring batch学习
  • 基础学习之——Docker 的基本概念和优势,以及在应用程序开发中的实际应用。
  • pycharm破解教程
  • 前端框架有哪些
  • 在VMware中的centos stream 9上用packstack安装openstack的单机版
  • 数字证书与公钥基础设施
  • 集成电路学习:什么是NOR Flash Memory非易失性闪存存储器
  • [译]前端离线指南(上)
  • AngularJS指令开发(1)——参数详解
  • Apache的基本使用
  • avalon2.2的VM生成过程
  • CentOS7简单部署NFS
  • Js实现点击查看全文(类似今日头条、知乎日报效果)
  • leetcode98. Validate Binary Search Tree
  • Otto开发初探——微服务依赖管理新利器
  • Python 反序列化安全问题(二)
  • 不上全站https的网站你们就等着被恶心死吧
  • 从零到一:用Phaser.js写意地开发小游戏(Chapter 3 - 加载游戏资源)
  • 从输入URL到页面加载发生了什么
  • 第2章 网络文档
  • 你不可错过的前端面试题(一)
  • 无服务器化是企业 IT 架构的未来吗?
  • 一个完整Java Web项目背后的密码
  • 再次简单明了总结flex布局,一看就懂...
  • 责任链模式的两种实现
  • JavaScript 新语法详解:Class 的私有属性与私有方法 ...
  • ​​​​​​​STM32通过SPI硬件读写W25Q64
  • ​Redis 实现计数器和限速器的
  • ​虚拟化系列介绍(十)
  • # Swust 12th acm 邀请赛# [ K ] 三角形判定 [题解]
  • #android不同版本废弃api,新api。
  • #AngularJS#$sce.trustAsResourceUrl
  • #ifdef 的技巧用法
  • %check_box% in rails :coditions={:has_many , :through}
  • (1)(1.8) MSP(MultiWii 串行协议)(4.1 版)
  • (6)添加vue-cookie
  • (C)一些题4
  • (JSP)EL——优化登录界面,获取对象,获取数据
  • (附源码)springboot课程在线考试系统 毕业设计 655127
  • (规划)24届春招和25届暑假实习路线准备规划
  • (十二)devops持续集成开发——jenkins的全局工具配置之sonar qube环境安装及配置
  • (数据大屏)(Hadoop)基于SSM框架的学院校友管理系统的设计与实现+文档
  • .NET Core 发展历程和版本迭代
  • .NET MVC、 WebAPI、 WebService【ws】、NVVM、WCF、Remoting
  • .NET牛人应该知道些什么(2):中级.NET开发人员
  • .stream().map与.stream().flatMap的使用
  • /deep/和 >>>以及 ::v-deep 三者的区别