软件设计模式系列之十一——装饰模式

news2025/4/16 7:55:51

当谈到设计软件系统时,经常需要考虑如何使系统更加灵活、可扩展和易维护。设计模式是一种被广泛采用的方法,用于解决常见的设计问题,并提供了一套可重用的解决方案。装饰模式(Decorator Pattern)是一种结构型设计模式,它允许您在不改变对象接口的情况下动态地添加对象的功能或责任。在本文中,我们将深入探讨装饰模式,包括其定义、举例说明、结构、实现步骤、代码实现、典型应用场景、优缺点、类似模式以及最后的小结。

1 模式的定义

装饰模式属于结构型设计模式,它通过将对象包装在装饰器类中来动态地添加额外的行为,而不需要修改原始对象的代码。这个模式以透明的方式向对象添加功能,从而使您可以根据需要组合各种功能。

它的主要目的是动态地给对象添加额外的功能,同时又不需要修改对象的代码。这一模式通过将对象包装在装饰器类中来实现功能的扩展,而不是通过继承。因此,装饰模式被称为一种替代继承的模式,因为它提供了一种比继承更加灵活的方式来扩展对象的功能。

装饰模式的核心思想是将对象的行为拆分为多个可组合的部分,每个部分都可以独立地扩展。装饰模式的关键概念是使用组合而不是继承来扩展对象的功能,从而避免了继承可能引发的类爆炸问题,并提供了更加灵活的方式来定制对象的行为。这种方式可以有效地应对需求变化,使得代码更具可扩展性和可维护性。

2 举例说明

让我们通过几个简单的示例来说明装饰模式的概念。
咖啡店的例子,假设我们有一个咖啡店,我们有一种基本的咖啡(SimpleCoffee)和一些可选的装饰品,如牛奶(MilkDecorator)和糖(SugarDecorator)。我们希望客户能够根据他们的口味自由选择添加装饰品,而不需要为每种可能的组合创建新的类。

衣着搭配的例子,在日常生活中,我们经常需要根据不同的场合选择不同的服装搭配。例如,一件基本的衬衫可以通过添加领带、领结、围巾、外套等装饰品来改变外观,而不需要改变衬衫本身。
在这里插入图片描述

餐厅点菜的例子,在餐厅用餐时,您可以根据口味选择不同的菜肴,并根据个人喜好添加调味品,如辣椒酱、酱油、芥末等。这些调味品可以看作是对菜肴的装饰,使您的餐点更加符合口味。

汽车定制的例子,汽车制造商通常提供多种基本型号的汽车,然后允许客户根据自己的需求和喜好添加各种选项和装饰品,如皮革座椅、音响系统、太阳顶等,以创建定制的汽车。

这些例子都展示了在日常生活中如何使用装饰模式来动态地扩展对象的功能,而无需修改原始对象的代码。这种模式使得我们可以根据需要定制和个性化物品,从而增加了灵活性和选择性。

3 结构

装饰模式的结构包括以下关键组件:
在这里插入图片描述

Component(抽象组件):定义了一个抽象接口,用于被具体组件和装饰器实现。在上面的示例中,Coffee 接口就是抽象组件。

ConcreteComponent(具体组件):实现了抽象组件的接口,是我们想要扩展功能的具体对象。在示例中,SimpleCoffee 就是具体组件。

Decorator(装饰器):抽象装饰器类,实现了抽象组件的接口,并包含一个对抽象组件的引用。这个类可以有一个或多个具体的装饰器子类。在示例中,MilkDecorator 和 SugarDecorator 就是装饰器。

ConcreteDecorator(具体装饰器):具体装饰器类扩展了装饰器,并添加了具体的功能。它们通常会调用父类的方法以保留原始功能,然后添加自己的功能。在示例中,MilkDecorator 和 SugarDecorator 分别是具体装饰器。

4 实现步骤

要实现装饰模式,您可以按照以下步骤进行操作:

创建一个抽象组件(Component),它定义了装饰器和具体组件的共同接口。
创建具体组件(ConcreteComponent),它是被装饰的对象,并实现了抽象组件的接口。
创建一个抽象装饰器(Decorator),它也实现了抽象组件的接口,并包含一个对抽象组件的引用。
创建具体装饰器(ConcreteDecorator),它扩展了抽象装饰器,并添加了具体的功能。
在客户端中,通过组合不同的具体组件和装饰器来创建对象,并调用其方法。

5 代码实现

以下是一个使用Java代码实现咖啡店点餐的装饰模式示例:
在这里插入图片描述

首先,我们定义一个抽象的咖啡接口 Coffee:

public interface Coffee {
    double getCost();
    String getDescription();
}

然后,创建具体的咖啡类 SimpleCoffee,它实现了 Coffee 接口:

public class SimpleCoffee implements Coffee {
    @Override
    public double getCost() {
        return 2.0;
    }

    @Override
    public String getDescription() {
        return "Simple Coffee";
    }
}

接下来,创建装饰器抽象类 CoffeeDecorator,它也实现了 Coffee 接口,并包含一个对抽象组件的引用:

public abstract class CoffeeDecorator implements Coffee {
    private final Coffee decoratedCoffee;

    public CoffeeDecorator(Coffee decoratedCoffee) {
        this.decoratedCoffee = decoratedCoffee;
    }

    @Override
    public double getCost() {
        return decoratedCoffee.getCost();
    }

    @Override
    public String getDescription() {
        return decoratedCoffee.getDescription();
    }
}

现在,我们可以创建具体的装饰器类,比如 MilkDecorator 和 SugarDecorator:

public class MilkDecorator extends CoffeeDecorator {
    public MilkDecorator(Coffee decoratedCoffee) {
        super(decoratedCoffee);
    }

    @Override
    public double getCost() {
        return super.getCost() + 1.0;
    }

    @Override
    public String getDescription() {
        return super.getDescription() + ", Milk";
    }
}

public class SugarDecorator extends CoffeeDecorator {
    public SugarDecorator(Coffee decoratedCoffee) {
        super(decoratedCoffee);
    }

    @Override
    public double getCost() {
        return super.getCost() + 0.5;
    }

    @Override
    public String getDescription() {
        return super.getDescription() + ", Sugar";
    }
}

现在,客户可以在咖啡店点餐并动态地添加装饰品:

public class CoffeeShop {
    public static void main(String[] args) {
        Coffee coffee = new SimpleCoffee();
        System.out.println("Cost: $" + coffee.getCost());
        System.out.println("Description: " + coffee.getDescription());

        // 添加牛奶和糖
        coffee = new MilkDecorator(coffee);
        coffee = new SugarDecorator(coffee);

        System.out.println("Cost: $" + coffee.getCost());
        System.out.println("Description: " + coffee.getDescription());
    }
}

这个示例演示了如何使用装饰模式来动态地扩展咖啡的功能,而不需要修改原始咖啡类。通过组合不同的装饰器,客户可以根据自己的口味点餐。

6 典型应用场景

装饰模式在许多情况下都有用,特别是当您需要在不修改现有代码的情况下扩展对象功能时。以下是一些典型的应用场景:
在这里插入图片描述

图形界面工具包:在GUI库中,装饰模式常用于添加额外的视觉效果,如边框、滚动条和工具提示,而无需修改原始组件的代码。

文件流处理:在文件处理中,您可以使用装饰模式来动态地添加压缩、加密、缓冲等功能,而不需要修改文件流的基本操作。

数据验证:在表单验证中,您可以使用装饰模式来添加各种验证规则,如必填字段、邮箱格式验证等,以提高代码的可维护性。

日志记录:装饰模式可以用于日志记录,使您能够动态地添加不同级别的日志信息,而不会影响原始业务逻辑。

7 优缺点

装饰模式具有以下优点和缺点:

优点:

  • 开闭原则。允许您添加新的装饰器类而不需要修改现有代码,遵守开闭原则(对扩展开放,对修改关闭)。

  • 灵活性。您可以根据需要组合不同的装饰器来创建复杂的对象,使系统更加灵活。

  • 单一职责原则。每个装饰器类都负责一个明确的功能,使得类的责任更加清晰。

  • 可重用性。由于装饰器可以独立使用,因此它们可以在不同的上下文中重复使用。

缺点:

  • 复杂性。如果使用不当,装饰器模式可能会导致类的层次结构变得复杂,使代码难以理解和维护。

  • 性能开销。每个装饰器都需要增加额外的开销,可能会影响性能,特别是在创建大量装饰对象时。

8 类似模式

有一些与装饰模式类似的设计模式,它们也关注于对象的功能扩展和组合,但在具体实现和应用上有一些不同。以下是一些与装饰模式相关的模式以及它们之间的联系。

适配器模式(Adapter Pattern):

适配器模式和装饰模式都属于结构型设计模式,它们都涉及到对象的包装。然而,它们的目的不同。适配器模式旨在兼容两个不同的接口,允许它们能够协同工作,而装饰模式旨在动态地添加功能,不改变原始接口。适配器模式涉及将一个接口转换成另一个接口,使得两者能够协同工作。装饰模式则是在不改变接口的前提下,动态地添加功能。在适配器模式中,适配器通常是一个新的类,而在装饰模式中,装饰器类与原始类共享相同的接口。

代理模式(Proxy Pattern):

代理模式和装饰模式都涉及到一个对象包装另一个对象。代理模式通常用于控制对对象的访问,例如,延迟加载、访问控制或监控。装饰模式用于动态地添加功能。代理模式的主要目的是控制访问,而装饰模式的主要目的是添加功能。代理通常在客户和真实对象之间充当中介,而装饰器是与真实对象共享相同接口的包装器。

组合模式(Composite Pattern):

组合模式和装饰模式都可以用于构建复杂的对象结构。它们都使用了递归组合对象,但目的不同。组合模式旨在创建树状结构,以表示部分-整体关系,并提供统一的方式来处理单个对象和组合对象。装饰模式用于动态地添加功能,通常是为了扩展单个对象的功能。

这些模式都与对象的功能扩展和组合有关,但它们的目的、用途和实现方式各不相同。装饰模式主要关注于动态添加功能而不改变接口,适配器模式关注于接口转换,代理模式关注于控制访问,组合模式关注于构建复杂的对象结构。在实际应用中,根据具体问题和需求,选择适合的设计模式是很重要的。

9 小结

装饰模式是一种强大的设计模式,它允许您在不修改现有代码的情况下动态地扩展对象的功能。通过定义抽象组件、具体组件、抽象装饰器和具体装饰器,您可以轻松地构建可维护和灵活的系统。然而,要小心不要过度使用装饰模式,以避免使代码变得复杂和难以理解。在适当的情况下,装饰模式可以成为您的设计工具箱中的强大工具,帮助您构建更加灵活和可扩展的软件系统。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

iOS应用上线需要注意的问题

将iOS应用上线到App Store需要仔细注意一系列问题,以确保应用的质量、安全性和用户体验。以下是一些在iOS应用上线过程中需要注意的关键问题,希望对大家有所帮助。北京木奇移动技术有限公司,专业的软件外包开发公司,欢迎交流合作。…

教你快速使用springboot整合图形验证码的两种方式

前言 今天给大家展示的是springboot使用图形验证码的两种方式,第一种基于hutool来实现,第二种方式基于axet实现。现在我们来谈一谈为什么要学习验证码 防止恶意攻击:验证码是一种常用的安全措施,它可以有效地防止恶意攻击&#x…

C++学习笔记——类与对象(六个默认成员函数)

1、构造函数 在一个类中,编译器会自动生成默认的成员函数,当对象进行初始化时,会默认调用这个函数来初始化。 构造函数是一个特殊的成员函数,名字与类名相同,创建类类型对象时由编译器自动调用,以保证 每个数据成员都有…

HTTPS的工作过程

HTTPS就是对HTTP进行了加密,因为要保证数据安全,就需要进行加密,网络中不再直接传输明文了,而是加密之后的密文,加密的方法有很多,但是整体可以分为两大类:对称加密和非对称加密 对称加密 对称加密其实就是…

Vue中的深度监听(Deep Watch):详细解析与实际示例

Vue中的深度监听(Deep Watch):详细解析与实际示例 Vue.js 是一款流行的前端 JavaScript 框架,其响应式系统是其核心特性之一。通过响应式系统,Vue允许开发者轻松地监听数据的变化并对其做出响应。在某些情况下&#x…

零基础学前端(七)将项目发布成网站

我们学习了HTML和CSS,已经可以做出精美的静态网页。我们不慌学习JavaScript,因为Javascript的作用是为网页增加动作和数据交换,只能让网页更完美而已,现在网页的基础我们已经可以搭建,我们不妨先将网站发布出去&#x…

uniapp选择地址弹窗组件

1.效果 2.子组件在components里面创建组件AddreessWindow <template><view style"position: relative;z-index: 999999 !important;"><view class"address-window" :class"value true ? on : "><view class"title…

Controller统一异常处理和yaml配置

目录 Controller统一异常处理 url解析 static下静态资源文件的访问 配置类 如何访问static下的资源文件 yaml基础语法 注解赋值 批量注入 单个注入 Controller统一异常处理 Controller统一异常处理ControllerAdvice&#xff1a;统一为Controller进行"增强" …

聊聊Spring中循环依赖与三级缓存

先看几个问题 什么事循环依赖&#xff1f;什么情况下循环依赖可以被处理&#xff1f;spring是如何解决循环依赖的&#xff1f; 什么是循环依赖&#xff1f; 简单理解就是实例 A 依赖实例 B 的同时 B 也依赖了 A Component public class A {// A 中依赖 BAutowiredprivate B b…

【表格插入小计行】el-table表格,数组对象中根据某字段插入小计行计算数据

前言 功能解释&#xff1a;遇到的一个需求&#xff0c;是表格的tabledata数组。里面有科室医生还有很多消费指标等数据。然后需要我排序后把科室放在一起。然后在每个科室下面添加一行数据&#xff0c;是小计行。用于计算上面相同科室的所有数据汇总。然后最下面再来个合计行&…

【深度学习实验】前馈神经网络(四):自定义逻辑回归模型:前向传播、反向传播算法

目录 一、实验介绍 二、实验环境 1. 配置虚拟环境 2. 库版本介绍 三、实验内容 0. 导入必要的工具包 1. 逻辑回归Logistic类 a. 构造函数__init__ b. __call__(self, x)方法 c. 前向传播forward d. 反向传播backward 2. 模型训练 3. 代码整合 一、实验介绍 实现逻…

JavaWeb 学习笔记 5:JSP

JavaWeb 学习笔记 5&#xff1a;JSP 简单的说&#xff0c;JSP 就是 Java Html&#xff0c;JSP 的出现是为了让 Java Web 应用生成动态页面更容易。 1.快速开始 1.1.依赖 添加 JSP 依赖&#xff1a; <dependency><groupId>javax.servlet.jsp</groupId>&…

华为云云耀云服务器L实例评测|使用docker部署禅道系统

大家好&#xff0c;我是早九晚十二&#xff0c;目前是做运维相关的工作。写博客是为了积累&#xff0c;希望大家一起进步&#xff01; 我的主页&#xff1a;早九晚十二 文章目录 前言准备工作华为云账号注册充值、购买服务器 服务器操作密码修改登录远程工具 禅道部署简介 部署…

【校招VIP】java语言考点之序列化

考点介绍&#xff1a; 将java对象转换为字节序列的过程称为对象的序列化。对象的序列化主要有两种用途: 1) 把对象的字节序列永久地保存到硬盘上&#xff0c;通常存放在一个文件中。 2) 在网络上传送对象的字节序列。 java语言考点之序列化-相关题目及解析内容可点击文章末尾链…

PyCharm:No Python interpreter configured for the project

一、问题概述 Your 的 Pycharm 软件创建完项目后&#xff0c;结果无法运行&#xff0c;观察后&#xff0c;在Pycharm代码编辑区上面出现了这样的一个黄色条提示&#xff1a;No Python interpreter configured for the project 【问题】在您的Python项目中无Python解释器…

金融业需要的大模型,是一个系统化工程

今年年初&#xff0c;在AIGC刚刚开始爆火的时候&#xff0c;我们曾经采访过一位AI领域的专家。当我们提问哪个行业将率先落地大模型时&#xff0c;他毫不犹豫地说道&#xff1a;“金融。” 金融行业场景多、数据多、知识多&#xff0c;这样的“三多”特点让其成为AI大模型发挥价…

yarn安装依赖时报错 error An unexpected error occurred:

一切起因是因为前一天安装了volta管理node&#xff0c;第二天启动项目&#xff0c; 显示error An unexpected error occurred: “https://registry.npmmirror.com/webpack-aliyun-oss/-/webpack-aliyun-oss-0.2.6.tgz: Request failed “404 Not Found””. 项目启动时发现报错…

Selenium Grid 的搭建方法

传统 Selenium Grid 的搭建方法 搭建一个具有 1 个 Node 的 Selenium Grid。那么通常来讲我们需要 2 台机器&#xff0c;其中一台作为 Hub&#xff0c;另外一台作为 Node&#xff0c;并要求这两台机器已经具备了 Java 执行环境。 1.通过官网下载 selenium-server-standalone-…

Java 21 发布,新功能助力开发更高效

Java 21 是 Java SE 平台的最新长期支持 (LTS) 版本&#xff0c;于 2023 年 9 月 19 日发布。它包括了一系列新功能和改进&#xff0c;可以让开发人员编写更高效、更可靠、更安全的 Java 应用程序。 新功能亮点 Java 21 的新功能包括&#xff1a; 虚拟线程&#xff1a;虚拟线程…