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

设计模式-结构型-装饰器模式

装饰器模式(Decorator Pattern)是结构型设计模式中的一种,它允许你通过将对象封装在一个新的对象中,来动态地添加新的功能,而无需改变原对象的结构。装饰器模式的核心思想是“将功能附加到对象上”,它是一种对象行为增强的模式。该模式通过组合而非继承来扩展对象的功能,这使得它比继承更灵活。

装饰器模式的定义

装饰器模式允许在不修改对象本身的情况下,动态地给一个对象添加额外的功能。其基本结构如下:

  • Component:定义一个对象接口,可以为其他对象提供基础功能。
  • ConcreteComponent:实现 Component 接口的具体类,是需要被装饰的原始对象。
  • Decorator:持有一个 Component 对象,并通过继承或实现该接口来增加附加功能。
  • ConcreteDecorator:具体的装饰类,它实现了装饰器接口,包装了 ConcreteComponent,并增加了新的功能。

装饰器模式的实现

在 Python 中,装饰器模式通常使用类继承组合的方式实现。为了更清晰地理解装饰器模式,我们可以通过一个示例进行讲解。

代码示例

假设我们有一个 Car 类,表示普通的汽车,车主希望能够动态地为这辆车添加新功能,例如导航系统和音响系统。我们可以通过装饰器模式为 Car 类动态地增加这些功能。

# 1. 基础组件接口
class Car:def features(self):raise NotImplementedError("Subclass must implement abstract method")# 2. 具体组件:原始的汽车类
class BasicCar(Car):def features(self):return "Basic Car Features"# 3. 装饰器类:持有一个 Car 对象
class CarDecorator(Car):def __init__(self, car):self._car = cardef features(self):return self._car.features()# 4. 具体装饰器:为汽车添加导航功能
class NavigationSystem(CarDecorator):def features(self):return f"{self._car.features()}, Navigation System"# 5. 具体装饰器:为汽车添加音响系统
class SoundSystem(CarDecorator):def features(self):return f"{self._car.features()}, Sound System"# 使用装饰器
basic_car = BasicCar()
print("Basic Car:", basic_car.features())# 添加导航功能
car_with_navigation = NavigationSystem(basic_car)
print("Car with Navigation:", car_with_navigation.features())# 添加音响功能
car_with_navigation_and_sound = SoundSystem(car_with_navigation)
print("Car with Navigation and Sound System:", car_with_navigation_and_sound.features())

输出结果

Basic Car: Basic Car Features

Car with Navigation: Basic Car Features, Navigation System

Car with Navigation and Sound System: Basic Car Features, Navigation System, Sound System

解释

  • BasicCar:原始汽车类,实现了 Car 接口,提供了基本的汽车功能。
  • CarDecorator:装饰器基类,它持有一个 Car 对象,并通过 features() 方法传递对原始 Car 功能的调用。
  • NavigationSystemSoundSystem:具体的装饰类,它们分别为汽车增加了导航和音响功能。

通过使用装饰器模式,我们可以动态地为 Car 添加新功能,而无需修改 BasicCar 类本身。

装饰器模式的优点

  1. 增强功能的灵活性

    • 通过装饰器模式,可以在运行时根据需求动态地为对象增加功能,而无需修改类的代码。它提供了一种非常灵活的方式来增强对象的行为。
  2. 符合开闭原则

    • 装饰器模式符合设计原则中的开闭原则:对扩展开放,对修改关闭。你可以通过装饰器为现有类添加新功能,而无需修改现有代码。
  3. 避免了继承层次的膨胀

    • 通过装饰器模式,我们不需要使用继承来增加新功能,从而避免了继承层次的复杂性。每个装饰类只负责一个功能,可以随意组合,避免了多重继承的繁琐。
  4. 组合与复用

    • 装饰器模式支持多重组合,可以灵活地为对象组合多个功能,每个功能都可以由不同的装饰器来处理,使得代码复用性更高。

装饰器模式的缺点

  1. 增加系统复杂度

    • 尽管装饰器模式提供了灵活的扩展性,但在实际使用时,过多的装饰器类可能导致系统变得复杂,难以管理和理解。
  2. 类数量激增

    • 每一个新的装饰类都可能成为系统中的一个新类。如果有很多功能需要动态添加,可能会导致类的数量剧增,增加系统的维护成本。
  3. 调试困难

    • 在使用装饰器模式时,调试可能会变得更加困难,因为装饰器通常是动态地改变对象的行为,跟踪这些变化可能需要更多的调试工具支持。

装饰器模式的实际应用

装饰器模式在许多实际应用中都有广泛的使用,特别是在需要动态增强对象功能的场景中。以下是一些典型应用场景:

  1. 输入输出流

    • 在 Java 中,装饰器模式常用于输入输出流的处理(例如:BufferedReaderFileReader)。每个类都为流对象添加不同的功能,而不需要修改原始对象的代码。
  2. GUI控件

    • 在图形用户界面(GUI)中,装饰器模式常用于为界面组件(如按钮、文本框)动态添加功能。例如,动态为按钮添加点击事件、为文本框添加验证功能等。
  3. Web开发中的请求处理

    • 在 Web 开发中,装饰器模式通常用于处理 HTTP 请求和响应,例如:在请求进入处理函数前动态地为请求添加身份验证、日志记录、权限控制等功能。
  4. 缓存和日志

    • 通过装饰器模式,可以在现有功能的基础上为数据处理添加缓存或日志功能,而不需要修改核心业务逻辑。

总结

装饰器模式是一种强大的设计模式,提供了一种灵活的方式来扩展对象的功能。通过将对象封装在另一个对象中,装饰器可以在不修改原始对象的情况下,为对象添加新的行为。这种模式使得我们可以通过组合多个装饰器来动态增加功能,而避免了类继承的复杂性,符合“开闭原则”。然而,过度使用装饰器模式可能导致系统复杂度和类数量的激增,因此需要谨慎使用。

相关文章:

设计模式-结构型-装饰器模式

装饰器模式(Decorator Pattern)是结构型设计模式中的一种,它允许你通过将对象封装在一个新的对象中,来动态地添加新的功能,而无需改变原对象的结构。装饰器模式的核心思想是“将功能附加到对象上”,它是一种…...

git详细使用教程

文章目录 一、 git介绍与安装1、git介绍2、git的安装3、git使用前的说明 二、git的基础使用1、走进git之前2、git基础使用1、git init 项目初始化(init)成仓库(repository)2、git add 管理文件3、git commit 把文件提交到仓库&…...

java实现word转html(支持docx及doc文件)

private final static String tempPath "C:\\Users\\xxx\\Desktop\\Word2Html\\src\\test\\";//图片及相关文件保存的路径public static void main(String argv[]) {try {JFileChooser fileChooser new JFileChooser();fileChooser.setDialogTitle("Select a …...

搜维尔科技:Xsens人形机器人解决方案的优势

Xsens 致力于推动人形机器人技术的发展,塑造机器人与人类环境无缝融合的未来,通过创新精确和协作,协助生产和服务,改善人类生活和产业。 Xsens通过人形跟随捕捉详细的人体运动数据,使机器人能够学习类人的动作&#x…...

【王树森搜索引擎技术】概要01:搜索引擎的基本概念

1. 基本名词 query:查询词SUG:搜索建议文档:搜索结果标签/筛选项 文档单列曝光 文档双列曝光 2. 曝光与点击 曝光:用户在搜索结果页上看到文档,就算曝光文档点击:在曝光后,用户点击文档&…...

《Java核心技术II》可中断套接字

4.2.4 可中断套接字 SocketChannel可以中断套接字 SocketChannel channel.open(new InetSocketAddress(host,port)); 通道(channel)并没有与之相关联的流,实际上,所拥有的read和write方法都是通过Buffer对象实现的。 如果不想处理缓冲区,…...

基于 Python 的深度学习的车俩特征分析系统,附源码

博主介绍:✌stormjun、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取源码联系🍅 👇🏻 精彩专栏推荐订阅👇&…...

C#读写ini配置文件保存设置参数

本示例使用设备:https://item.taobao.com/item.htm?spma21dvs.23580594.0.0.52de2c1b5P5rkA&ftt&id22173428704 [DllImport("kernel32", CharSet CharSet.Unicode)] public static extern uint GetPrivateProfileString(string lpAppName, stri…...

SwanLab环境变量列表

文章目录 环境变量全局配置服务配置登录认证其他 环境变量 ⚙️完整环境变量1 -> Github、⚙️完整环境变量2 -> Github 全局配置 环境变量描述默认值SWANLAB_SAVE_DIRSwanLab 全局文件夹保存的路径用户主目录下的 .swanlab 文件夹SWANLAB_LOG_DIRSwanLab 解析日志文件…...

深度学习入门-CNN

一、CNN是什么 CNN,即卷积神经网络(convolutional neural network),是用于预测的标准神经网络架构。在人工智能的广阔领域中,CNN被用于图像识别、语音识别等各种场合,CNN通过模拟人类视觉皮层的神经元连接方…...

微服务网关,如何选择?

什么是API网关 API网关(API Gateway)是微服务架构中的一个关键组件,它充当了客户端与后端服务之间的中间层。其主要功能包括请求路由、协议转换、负载均衡、安全认证、限流熔断等。通过API网关,客户端无需直接与多个微服务交互&a…...

SpringBoot集成Mqtt服务实现消费发布和接收消费

该项目介绍了docker环境下如何安装mqtt和springboot集成mqtt服务 前述 MQTT(Message Queuing Telemetry Transport)是一种轻量级的消息传输协议,设计用于在资源受限的设备和低带宽、不可靠的网络连接中高效地传输数据。主要用于物联网设备传输,设备之间可以高效地交换数据…...

在Mac mini上实现本地话部署AI和知识库

在Mac mini上实现本地话部署AI和知识库 硬件要求:大模型AI,也叫LLM,需要硬件支持,常见的方式有2种:一种是采用英伟达之类支持CUDA库的GPU芯片或者专用AI芯片;第二种是采用苹果M系列芯片架构的支持统一内存架…...

一个方法被多个线程同时调用,确保同样参数的调用只能有一个线程执行,不同参数的调用则可以多个线程同时执行

我们知道通过lock一个固定静态object给代码段加同步锁,可以让多个线程的同时调用以同步执行,因此可以利用字典来给不同参数分配不同的静态对象,方法中不同的参数调用锁住各自不同的静态对象即可实现不同参数不加锁,相同参数才加锁…...

3. MySQL事务并发的问题与解决方法

一. 并发事务带来的问题 并发会造成事务间出现脏读,不可重复读,幻读现象。 1. 脏读 一个事务在处理过程中读取了另外一个事务未提交的数据。若另外一个事务回滚,则读取到的数据是无效的,又称为脏读。 2. 不可重复读 在一个事务…...

25/1/15 嵌入式笔记 初学STM32F108

GPIO初始化函数 GPIO_Ini:初始化GPIO引脚的模式,速度和引脚号 GPIO_Init(GPIOA, &GPIO_InitStruct); // 初始化GPIOA的引脚0 GPIO输出控制函数 GPIO_SetBits:将指定的GPIO引脚设置为高电平 GPIO_SetBits(GPIOA, GPIO_Pin_0); // 将GPIO…...

MySQL的不同SQL模式导致行为不同?

现象: 我在两个mysql库都有相同定义的表,其中一个字段是varchar(1200)。当我都对这个表进行insert操作,而且超过此字段的规定长度(此处是1200),这两库的行为是不一样的:库B是直接报错too long&…...

Flink 使用 Kafka 作为数据源时遇到了偏移量提交失败的问题

具体的错误日志 21:43:57.069 [Kafka Fetcher for Source: Custom Source -> Map -> Filter (1/1)#2] ERROR org.apache.kafka.clients.consumer.internals.ConsumerCoordinator - [Consumer clientIdconsumer-my-group-6, groupIdmy-group] Offset commit failed on pa…...

【日志篇】(7.6) ❀ 01. 在macOS下刷新FortiAnalyzer固件 ❀ FortiAnalyzer 日志分析

【简介】FortiAnalyzer 是 Fortinet Security Fabric 安全架构的基础,提供集中日志记录和分析,以及端到端可见性。因此,分析师可以更有效地管理安全状态,将安全流程自动化,并快速响应威胁。具有分析和自动化功能的集成…...

LSA更新、撤销

LSA的新旧判断&#xff1a; 1.seq&#xff0c;值越大越优先 2.chksum&#xff0c;值越大越优先 3.age&#xff0c;本地的LSA age和收到的LSA age作比较 如果差值<900s&#xff0c;认为age一致&#xff0c;保留本地的&#xff1a;我本地有一条LSA是100 你给的是400 差值小于…...

基于Flask实现的医疗保险欺诈识别监测模型

基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施&#xff0c;由雇主和个人按一定比例缴纳保险费&#xff0c;建立社会医疗保险基金&#xff0c;支付雇员医疗费用的一种医疗保险制度&#xff0c; 它是促进社会文明和进步的…...

理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端

&#x1f31f; 什么是 MCP&#xff1f; 模型控制协议 (MCP) 是一种创新的协议&#xff0c;旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议&#xff0c;它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...

汽车生产虚拟实训中的技能提升与生产优化​

在制造业蓬勃发展的大背景下&#xff0c;虚拟教学实训宛如一颗璀璨的新星&#xff0c;正发挥着不可或缺且日益凸显的关键作用&#xff0c;源源不断地为企业的稳健前行与创新发展注入磅礴强大的动力。就以汽车制造企业这一极具代表性的行业主体为例&#xff0c;汽车生产线上各类…...

React19源码系列之 事件插件系统

事件类别 事件类型 定义 文档 Event Event 接口表示在 EventTarget 上出现的事件。 Event - Web API | MDN UIEvent UIEvent 接口表示简单的用户界面事件。 UIEvent - Web API | MDN KeyboardEvent KeyboardEvent 对象描述了用户与键盘的交互。 KeyboardEvent - Web…...

【RockeMQ】第2节|RocketMQ快速实战以及核⼼概念详解(二)

升级Dledger高可用集群 一、主从架构的不足与Dledger的定位 主从架构缺陷 数据备份依赖Slave节点&#xff0c;但无自动故障转移能力&#xff0c;Master宕机后需人工切换&#xff0c;期间消息可能无法读取。Slave仅存储数据&#xff0c;无法主动升级为Master响应请求&#xff…...

Linux C语言网络编程详细入门教程:如何一步步实现TCP服务端与客户端通信

文章目录 Linux C语言网络编程详细入门教程&#xff1a;如何一步步实现TCP服务端与客户端通信前言一、网络通信基础概念二、服务端与客户端的完整流程图解三、每一步的详细讲解和代码示例1. 创建Socket&#xff08;服务端和客户端都要&#xff09;2. 绑定本地地址和端口&#x…...

安全突围:重塑内生安全体系:齐向东在2025年BCS大会的演讲

文章目录 前言第一部分&#xff1a;体系力量是突围之钥第一重困境是体系思想落地不畅。第二重困境是大小体系融合瓶颈。第三重困境是“小体系”运营梗阻。 第二部分&#xff1a;体系矛盾是突围之障一是数据孤岛的障碍。二是投入不足的障碍。三是新旧兼容难的障碍。 第三部分&am…...

十九、【用户管理与权限 - 篇一】后端基础:用户列表与角色模型的初步构建

【用户管理与权限 - 篇一】后端基础:用户列表与角色模型的初步构建 前言准备工作第一部分:回顾 Django 内置的 `User` 模型第二部分:设计并创建 `Role` 和 `UserProfile` 模型第三部分:创建 Serializers第四部分:创建 ViewSets第五部分:注册 API 路由第六部分:后端初步测…...

AxureRP-Pro-Beta-Setup_114413.exe (6.0.0.2887)

Name&#xff1a;3ddown Serial&#xff1a;FiCGEezgdGoYILo8U/2MFyCWj0jZoJc/sziRRj2/ENvtEq7w1RH97k5MWctqVHA 注册用户名&#xff1a;Axure 序列号&#xff1a;8t3Yk/zu4cX601/seX6wBZgYRVj/lkC2PICCdO4sFKCCLx8mcCnccoylVb40lP...

rm视觉学习1-自瞄部分

首先先感谢中南大学的开源&#xff0c;提供了很全面的思路&#xff0c;减少了很多基础性的开发研究 我看的阅读的是中南大学FYT战队开源视觉代码 链接&#xff1a;https://github.com/CSU-FYT-Vision/FYT2024_vision.git 1.框架&#xff1a; 代码框架结构&#xff1a;readme有…...