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

ops/2024/10/21 9:24:17/

一、什么是装饰器模式

    装饰器模式(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方法来分别描述咖啡和计算价格:

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

6.2. 创建具体组件

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

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

6.3. 创建装饰器角色

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

java">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并添加自己特有的功能(如描述和价格)。

java">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();}
}
java">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. 创建客户端测试制作咖啡

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

java">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())),并打印了它的描述和价格。这样,我们就可以看到装饰器模式如何动态地给对象添加额外的职责。

七、总结

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

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


http://www.ppmy.cn/ops/109778.html

相关文章

Android系列基础知识总结

四大组件 Activity Activity生命周期 不同场景下Activity生命周期的变化过程 启动Activity: onCreate()—>onStart()—>onResume(),Activity进入运行状态。Activity退居后台: 当前Activity转到新的Activity界面或按Home键回到主屏&a…

HTTP 协议的工作过程

当我们在浏览器输入一个网址,此时浏览器就会给对应的服务器发送一个 HTTP 请求,对应的服务器收到这个请求之后,经过计算处理,就会返回一个 HTTP 响应。并且当我们访问一个网站时,可能涉及不止一次的 HTTP 请求和响应的…

电气设备或电气线路故障引起的火灾主要特征

1 主要特征 电气火灾是由电气设备或电气线路故障引起的火灾。这类火灾具有一定的特殊性,主要特征如下: 1)突发性强 突发性:电气火灾往往在没有明显征兆的情况下突然发生,不易被及时察觉。 瞬间爆发:由于…

Vue.js 计算属性

Vue.js 计算属性 Vue.js 是一款流行的前端JavaScript框架,以其简洁、灵活和高效的特点著称。在Vue.js中,计算属性(computed properties)是一种核心特性,它允许开发者定义依赖于其他数据的复杂逻辑,并且能够…

【H2O2|全栈】关于CSS(1)CSS基础(一)

目录 CSS基础知识 前言 准备工作 啥是CSS? 如何引用CSS? 选择器 通配符选择器 类名(class)选择器 id选择器 CSS解析顺序(优先级) 常见CSS标签(一) 字体属性 font-style…

CocosCreator面试真题详解

最近有位同学面试Cocos Creator,我们把面试时问道的真题列举出来,并配上参考答案。 问题1: 你们公司项目时如何做战斗系统的? 面试官你好,做战斗系统和架构的时候,我们一般把代码逻辑分成3层来设计,同时把数据独立出…

solidity从入门到精通(持续更新)

我一度觉得自己不知何时变成了一个浮躁的人,一度不想受外界干扰的我被干扰了,再无法平静的去看一本书,但我仍旧希望我能够克服这些,压抑着自己直到所有的冲动和奇怪的思想都无法再左右我行动。 自律会让你更加自律,放纵…

Vue Echarts报错Initialize failed: invalid dom解决方法

此问题是图表初始化时 找不到dom,以下是解决方法 1、不要用created(用mounted),created这时候还只是创建了实例,但模板还没挂载完成; created: 在模板渲染成 html 前调用,通常初始…