【Python笔记-设计模式】装饰器模式

news/2024/6/22 10:03:38

一、说明

装饰器模式是一种结构型设计模式,旨在动态的给一个对象添加额外的职责。

(一) 解决问题

不改变原有对象结构的情况下,动态地给对象添加新的功能或职责,实现透明地对对象进行功能的扩展。

(二) 使用场景

  • 如果用继承来扩展对象行为的方案难以实现或者根本不可行时,可以考虑使用装饰器模式。
  • 希望在无需修改代码的情况下即可使用对象, 且希望在运行时为对象新增额外的行为。

二、结构

  1. 部件(Component)声明封装器和被封装对象的公用接口。
  2. 具体部件(Concrete Component)类是被封装对象所属的类。它定义了基础行为,但装饰类可以改变这些行为。
  3. 基础装饰(Base Decorator)类拥有一个指向被封装对象的引用成员变量。该变量的类型应当被声明为通用部件接口,这样它就可以引用具体的部件和装饰。装饰基类会将所有操作委派给被封装的对象。
  4. 具体装饰类(Concrete Decorators)定义了可动态添加到部件的额外行为。具体装饰类会重写装饰基类的方法,并在调用父类方法之前或之后进行额外的行为。
  5. 客户端(Client)可以使用多层装饰来封装部件,只要它能使用通用接口与所有对象互动即可。

三、伪代码

#!/usr/bin/env python
# -*- coding: UTF-8 -*-
__doc__ = """
装饰器模式例:实现咖啡店的订单系统,不同种类的咖啡可以搭配不同的调料,使用装饰器对原有对象 增加相应配料信息和价格
"""class Coffee:"""基础组件类 - 咖啡"""def cost(self):return 9.9class Espresso(Coffee):"""具体组件类 - 浓缩咖啡"""def cost(self):return super().cost() + 1def description(self):return "浓缩咖啡"class Decorator(Coffee):"""装饰器基类 - 调料"""def __init__(self, coffee):self._coffee = coffeedef cost(self):return self._coffee.cost()def description(self):return self._coffee.description()class Milk(Decorator):"""具体装饰器类 - 牛奶"""def cost(self):return super().cost() + 2def description(self):return super().description() + " + 牛奶"class Sugar(Decorator):"""具体装饰器类 - 糖"""def cost(self):return super().cost() + 0.5def description(self):return super().description() + " + 糖"# 客户端代码
if __name__ == "__main__":"""浓缩咖啡 (10.9 💰)浓缩咖啡 + 牛奶 (12.9 💰)浓缩咖啡 + 牛奶 + 糖 (13.4 💰)"""coffee = Espresso()print(f"{coffee.description()} ({coffee.cost()} 💰)")# 加入牛奶coffee_with_milk = Milk(coffee)print(f"{coffee_with_milk.description()} ({coffee_with_milk.cost()} 💰)")# 再加入糖coffee_with_sugar = Sugar(coffee_with_milk)print(f"{coffee_with_sugar.description()} ({coffee_with_sugar.cost()} 💰)")

四、优缺点

优点

  • 灵活性高:装饰器模式可以动态地给一个对象添加额外的功能,而不需要改变其结构
  • 单一职责原则。 你可以将实现了许多不同行为的一个大类拆分为多个较小的类。
  • 提高了代码复用性:可以将装饰器类与具体的组件类分开,使得装饰器类可以复用于不同的组件类。

缺点

  • 顺序对功能影响:装饰器的顺序会影响最终的功能效果,可能会导致功能组合的结果不可预测,需要谨慎设计装饰器的顺序。
  • 增加了系统复杂性

 跳转主页:【Python笔记】设计模式-CSDN博客

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

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

相关文章

07 MyBatis之高级映射 + 懒加载(延迟加载)+缓存

1. 高级映射 例如有两张表, 分别为班级表和学生表 自然, 一个班级对应多个学生 像这种数据 , 应该如果如何映射到Java的实体类上呢? 这就是高级映射解决的问题 以班级和学生为例子 , 因为一个班级对应多个学生 , 因此学生表中必定有一个班级编号字段cid 但我们在学生的实体…

【风格迁移】CAST:对比学习,从图像特征而非其二阶统计量(Gram矩阵)中学习风格

CAST:对比学习,从图像特征而非其二阶统计量(Gram矩阵)中学习风格 提出背景5 why 分析5 so分析 CAST 框架多层风格投影器领域增强模块生成网络 效果对比 StyleGAN 提出背景 论文:https://arxiv.org/pdf/2205.09542.pdf…

Clickhouse的安装

1.准备工作 确定防火墙处于关闭状态 ulimit -a 2.CentOS取消打开文件数限制 (1)在 hadoop102 的 /etc/security/limits.conf 文件的末尾加入以下内容 sudo vim /etc/security/limits.conf * soft nofile 65536 * hard nofile 65536 * soft nproc …

Sora - 探索AI视频模型的无限可能

随着人工智能技术的飞速发展,AI视频模型已成为科技领域的新热点。而在这个浪潮中,OpenAI推出的首个AI视频模型Sora,以其卓越的性能和前瞻性的技术,引领着AI视频领域的创新发展。让我们将一起探讨Sora的技术特点、应用场景以及对未…

LeetCode206: 反转链表.

题目描述 给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。 示例 解题方法 假设链表为 1→2→3→∅,我们想要把它改成∅←1←2←3。在遍历链表时,将当前节点的 next指针改为指向前一个节点。由于节点没有引用其前一…

挑战30天学完Python:Day16 日期时间

📘 Day 16 🎉 本系列为Python基础学习,原稿来源于 30-Days-Of-Python 英文项目,大奇主要是对其本地化翻译、逐条验证和补充,想通过30天完成正儿八经的系统化实践。此系列适合零基础同学,或仅了解Python一点…