设计模式 15 Decorator Pattern 装饰器模式

2024-05-28 06:12

本文主要是介绍设计模式 15 Decorator Pattern 装饰器模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

设计模式 15 Decorator Pattern 装饰器模式
1.定义

Decorator Pattern 装饰器模式是一种结构型设计模式,它允许在运行时给对象添加新的行为或职责,而无需修改对象的源代码。这种模式通过创建一个包装对象,也称为装饰器,来包裹原始对象,装饰器对象与原始对象有相同的接口,因此可以在不改变客户端代码的情况下,增加或修改对象的功能。

装饰器模式的优点包括:

动态地给对象添加新的行为,而无需修改对象的源代码或继承结构。
可以独立地增加对象的功能,因为每个装饰器都是独立的类。
保持了类的单一职责,使得代码更易于维护和扩展。
装饰器模式通常用于添加非核心功能,如日志、性能追踪、缓存等,而不会影响对象的核心行为。


2.内涵


 Decorator Pattern 的主要组成部分:

  • Component(组件):这是定义对象接口的抽象类或接口。所有可以装饰的对象都必须实现这个接口,这样装饰器才能与它们互换。
  • Concrete Component(具体组件):这是 Component 接口的实现,是将要被装饰的对象。它定义了实际的行为和职责。
  • Decorator(装饰器):这是 Component 接口的实现,它持有一个 Component 对象的引用。装饰器可以是抽象的,也可以包含具体的行为。装饰器对象可以添加新的行为或修改 Component 对象的行为。
  • Concrete Decorator(具体装饰器):这是 Decorator 的具体实现,它给 Component 添加新的行为或职责。每个 Concrete Decorator 都可以添加不同的功能,也可以堆叠多个 Decorator 来增强对象的功能。

组件之间的调用图如下图所示:

+------------+| Component  |+------------+|| 继承/实现V+--------------+|  Decorator   |+--------------+|| 继承/实现V+-------------------+| Concrete Decorator|+-------------------+|| 持有V+-------------------+| Concrete Component|+-------------------+


每个模块的作用如下:

  • Component Interface:定义了公共接口,使得装饰器和组件可以互相协作。
  • Concrete Component:实现了 Component 接口,定义了具体的行为和状态,是被装饰的对象。
  • Decorator:作为抽象装饰器,持有 Component 的引用,实现 Component 接口,以保持与组件的兼容性。
  • Concrete Decorator:具体实现了装饰器的逻辑,添加或修改了 Concrete Component 的行为。可以有多个 Concrete Decorator,每个实现不同的增强功能。

3.使用示例

#include <iostream>
#include <string>using namespace std;// Component interface - defines the basic ice cream
// operations.
class IceCream {
public:virtual string getDescription() const = 0;virtual double cost() const = 0;
};// Concrete Component - the basic ice cream class.
class VanillaIceCream : public IceCream {
public:string getDescription() const override{return "Vanilla Ice Cream";}double cost() const override { return 160.0; }
};// Decorator - abstract class that extends IceCream.
class IceCreamDecorator : public IceCream {
protected:IceCream* iceCream;public:IceCreamDecorator(IceCream* ic): iceCream(ic){}string getDescription() const override{return iceCream->getDescription();}double cost() const override{return iceCream->cost();}
};// Concrete Decorator - adds chocolate topping.
class ChocolateDecorator : public IceCreamDecorator {
public:ChocolateDecorator(IceCream* ic): IceCreamDecorator(ic){}string getDescription() const override{return iceCream->getDescription()+ " with Chocolate";}double cost() const override{return iceCream->cost() + 100.0;}
};// Concrete Decorator - adds caramel topping.
class CaramelDecorator : public IceCreamDecorator {
public:CaramelDecorator(IceCream* ic): IceCreamDecorator(ic){}string getDescription() const override{return iceCream->getDescription() + " with Caramel";}double cost() const override{return iceCream->cost() + 150.0;}
};// 测试案例分析调用
int main()
{// Create a vanilla ice creamIceCream* vanillaIceCream = new VanillaIceCream();cout << "Order: " << vanillaIceCream->getDescription()<< ", Cost: Rs." << vanillaIceCream->cost()<< endl;// Wrap it with ChocolateDecoratorIceCream* chocolateIceCream= new ChocolateDecorator(vanillaIceCream);cout << "Order: " << chocolateIceCream->getDescription()<< ", Cost: Rs." << chocolateIceCream->cost()<< endl;// Wrap it with CaramelDecoratorIceCream* caramelIceCream= new CaramelDecorator(chocolateIceCream);cout << "Order: " << caramelIceCream->getDescription()<< ", Cost: Rs." << caramelIceCream->cost()<< endl;delete vanillaIceCream;delete chocolateIceCream;delete caramelIceCream;return 0;
}

4.注意事项


在使用 Decorator Pattern 时,需要注意以下几点:

  • 性能影响:装饰器可能会增加对象的创建和管理成本,特别是在需要大量创建和销毁对象的场景下。因此,需要权衡装饰器带来的灵活性和可能的性能损失。
  • 代码复杂性:如果过度使用装饰器,可能会导致代码结构变得复杂,难以理解和维护。确保每个装饰器都有明确的职责,并保持代码的简洁性。
  • 类型检查和强类型语言:在强类型语言中,装饰器可能会隐藏原始对象的类型,这可能导致类型检查问题。使用类型注解或接口可以帮助解决这个问题。
  • 一致性:确保所有装饰器的行为与组件接口保持一致,否则可能会导致客户端代码出错或行为不一致。
  • 可组合性:虽然装饰器可以堆叠,但过多的装饰器可能导致代码难以理解和调试。考虑使用组合模式来组合多个功能,而不是一次性添加多个装饰器。
  • 状态管理:如果组件的状态对行为有影响,确保装饰器正确处理和传递这些状态,以避免意外的行为。
  • 设计时的考虑:在设计系统时,提前考虑是否需要使用装饰器,因为它可能影响到类的设计和接口的定义。在开始编码之前,充分理解需求和扩展性要求,以便做出最佳决策。
5.最佳实践


该模式,最佳实践包括以下这些点:

  • 保持装饰器和组件接口一致:装饰器应该与组件有相同的接口,这样客户端代码可以透明地使用装饰后的对象,而无需知道它是装饰器还是原始组件。
  • 避免深度装饰:虽然可以堆叠多个装饰器,但过多的装饰可能导致代码复杂性增加。如果需要添加大量功能,可能需要考虑其他设计模式,如组合模式或使用类的继承。
  • 使用接口而非具体类:装饰器模式通常与接口一起使用,因为接口允许更灵活的替换和扩展。如果使用具体类,可能会限制装饰器的通用性。
  • 明确职责:每个装饰器应专注于添加或修改特定的行为,而不是试图一次性处理所有额外功能。这样可以保持代码的清晰和可维护性。
  • 使用装饰器来扩展功能:装饰器模式最适合用于添加非核心功能,如日志、缓存、权限控制等,这些功能可以独立于核心业务逻辑存在。
  • 避免与继承混淆:装饰器模式是作为继承的替代方案,特别是当需要动态地添加或移除行为时。如果新的行为是静态的,并且适用于所有对象,那么继承可能更合适。
  • 测试和文档:确保为装饰器编写测试用例,并在文档中明确说明装饰器的作用,以便其他开发者理解其功能和使用方式。
6.总结

该模式在使用时可能存在以下“坑”:类型混淆:装饰器可能会隐藏原始对象的类型,导致类型检查问题。例如,在强类型语言中,如果装饰器没有正确地保持原始类型信息,可能会在编译时或运行时遇到错误。例如,Java 中的 InputStream 和其装饰器,如果不注意类型转换,可能会导致类型安全问题。此外,性能开销也是需要考虑的地方。


这篇关于设计模式 15 Decorator Pattern 装饰器模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1009751

相关文章

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

模版方法模式template method

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/template-method 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤。 上层接口有默认实现的方法和子类需要自己实现的方法

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

迭代器模式iterator

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/iterator 不暴露集合底层表现形式 (列表、 栈和树等) 的情况下遍历集合中所有的元素

《x86汇编语言:从实模式到保护模式》视频来了

《x86汇编语言:从实模式到保护模式》视频来了 很多朋友留言,说我的专栏《x86汇编语言:从实模式到保护模式》写得很详细,还有的朋友希望我能写得更细,最好是覆盖全书的所有章节。 毕竟我不是作者,只有作者的解读才是最权威的。 当初我学习这本书的时候,只能靠自己摸索,网上搜不到什么好资源。 如果你正在学这本书或者汇编语言,那你有福气了。 本书作者李忠老师,以此书为蓝本,录制了全套视频。 试

利用命令模式构建高效的手游后端架构

在现代手游开发中,后端架构的设计对于支持高并发、快速迭代和复杂游戏逻辑至关重要。命令模式作为一种行为设计模式,可以有效地解耦请求的发起者与接收者,提升系统的可维护性和扩展性。本文将深入探讨如何利用命令模式构建一个强大且灵活的手游后端架构。 1. 命令模式的概念与优势 命令模式通过将请求封装为对象,使得请求的发起者和接收者之间的耦合度降低。这种模式的主要优势包括: 解耦请求发起者与处理者

springboot实战学习(1)(开发模式与环境)

目录 一、实战学习的引言 (1)前后端的大致学习模块 (2)后端 (3)前端 二、开发模式 一、实战学习的引言 (1)前后端的大致学习模块 (2)后端 Validation:做参数校验Mybatis:做数据库的操作Redis:做缓存Junit:单元测试项目部署:springboot项目部署相关的知识 (3)前端 Vite:Vue项目的脚手架Router:路由Pina:状态管理Eleme