设计模式 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 cream
    IceCream* vanillaIceCream = new VanillaIceCream();
    cout << "Order: " << vanillaIceCream->getDescription()
        << ", Cost: Rs." << vanillaIceCream->cost()
        << endl;

    // Wrap it with ChocolateDecorator
    IceCream* chocolateIceCream
        = new ChocolateDecorator(vanillaIceCream);
    cout << "Order: " << chocolateIceCream->getDescription()
        << ", Cost: Rs." << chocolateIceCream->cost()
        << endl;

    // Wrap it with CaramelDecorator
    IceCream* 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 和其装饰器,如果不注意类型转换,可能会导致类型安全问题。此外,性能开销也是需要考虑的地方。

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

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

相关文章

Ubuntu Nerfstudio安装

https://blog.csdn.net/qq_30565883/article/details/133778529 https://blog.csdn.net/weixin_52581013/article/details/137982846 https://zhuanlan.zhihu.com/p/654394767 1. 结论 因为需要安装tiny-cuda-nn&#xff0c;然而 所以我之前的在笔记本上安装就白费了&#xf…

基于python的k-means聚类分析算法,对文本、数据等进行聚类,有轮廓系数和手肘法检验

K-means算法是一种常见的聚类算法&#xff0c;用于将数据点分成不同的组&#xff08;簇&#xff09;&#xff0c;使同一组内的数据点彼此相似&#xff0c;不同组之间的数据点相对较远。以下是K-means算法的基本工作原理和步骤&#xff1a; 工作原理&#xff1a; 初始化&#x…

QT C++ QTableWidget 演示

本文演示了 QTableWidget的初始化以及单元格值改变时响应槽函数&#xff0c;打印单元格。 并且&#xff0c;最后列不一样,是combobox &#xff0c;此列的槽函数用lambda函数。 在QT6.2.4 MSVC2019 调试通过。 1.界面效果 2.头文件 #ifndef MAINWINDOW_H #define MAINWINDOW…

使用API有效率地管理Dynadot域名,进行域名邮箱的默认邮件转发设置

关于Dynadot Dynadot是通过ICANN认证的域名注册商&#xff0c;自2002年成立以来&#xff0c;服务于全球108个国家和地区的客户&#xff0c;为数以万计的客户提供简洁&#xff0c;优惠&#xff0c;安全的域名注册以及管理服务。 Dynadot平台操作教程索引&#xff08;包括域名邮…

【四数之和】python,排序+双指针

四层循环&#xff1f;&#xff08;doge) 和【三数之和】题目很类似 class Solution:def fourSum(self, nums: List[int], target: int) -> List[List[int]]:nums.sort()#a,b,c,d四个数&#xff0c;先固定两个数&#xff0c;那就是双指针问题了&#xff0c;令ba1&#xff…

【数据结构】【C语言】堆~动画超详细解读!

目录 1 什么是堆1.1 堆的逻辑结构和物理结构1.2 堆的访问1.3 堆为什么物理结构上要用数组?1.4 堆数据上的特点 2 堆的实现2.1 堆类型定义2.2 需要实现的接口2.3 初始化堆2.4 销毁堆2.5 堆判空2.6 交换函数2.7 向上调整(小堆)2.8 向下调整(小堆)2.9 堆插入2.10 堆删除2.11 //堆…

若依解决使用https上传文件返回http路径问题

若依通过HTTPS请求进行文件上传时却返回HTTP的文件链接地址&#xff0c;主要原因是使用了 request.getRequestURL 获取链接地址。 解决办法&#xff1a; 在nginx配置文件location处加上&#xff1a;proxy_set_header X-Forwarded-Scheme $scheme; 然后代码通过request.getHea…

【跳坑日记】暴力解决Ubuntu SSH报错: Failed to start OpenBSD Secure Shell server

报错环境说明&#xff1a; 服务器环境&#xff1a;Ubuntu 20.04 错误内容 最近服务器突然报错&#xff0c;提示如下图信息&#xff1a; 搜素了各种问答&#xff0c;国内的回答大多数是用 ssh-keygen -A命令来解决&#xff0c;但最终也无法登录服务器。 最终搜索到ask ubun…

比较kube-proxy模式:iptables还是IPVS?

kube-proxy是任何 Kubernetes 部署中的关键组件。它的作用是将流向服务&#xff08;通过集群 IP 和节点端口&#xff09;的流量负载均衡到正确的后端pod。kube-proxy可以运行在三种模式之一&#xff0c;每种模式都使用不同的数据平面技术来实现&#xff1a;userspace、iptables…

go-zero 实战(3)

引入 Redis 在之前的 user 微服务中引入 redis。 1. 修改 user/internal/config/config.go package configimport ("github.com/zeromicro/go-zero/core/stores/cache""github.com/zeromicro/go-zero/zrpc" )type Config struct {zrpc.RpcServerConfMys…

代码随想录算法训练营第36期DAY35

DAY35 122买卖股票的最佳时机ii 很巧妙&#xff0c;也很难想到&#xff1a;计算每天的利润&#xff08;今天卖出&#xff0c;昨天买入的利润&#xff09;&#xff0c;只取正数相加。 class Solution {public: int maxProfit(vector<int>& prices) { int…

【机器学习300问】93、到底什么是优化器optimizer?

本文是对之前我写的梯度下降优化算法相关内容进行一次简要总结。在学习PyTorch框架的过程中&#xff0c;会遇到“优化器”&#xff08;optimizer&#xff09;这个概念。我想用通俗易懂的方式&#xff0c;说说优化器到底是个什么东西&#xff0c;并在此基础上&#xff0c;将前文…

Qt代码初识

文章目录 Qt代码初识1. Qt Hello World 程序1.1 使⽤ "按钮" 实现1.1.1 纯代码⽅式实现1.1.2 可视化操作实现 1.2 使⽤ "标签" 实现1.2.1 纯代码⽅式实现1.2.2 可视化操作实现 2. 项⽬⽂件解析2.1 .pro ⽂件解析2.2 widget.h ⽂件解析2.3 main.cpp ⽂件解析…

SwanLab入门深度学习:BERT IMDB文本情感分类

基于BERT模型的IMDB电影评论情感分类&#xff0c;是NLP经典的Hello World任务之一。 这篇文章我将带大家使用SwanLab、transformers、datasets三个开源工具&#xff0c;完成从数据集准备、代码编写、可视化训练的全过程。 观察了一下&#xff0c;中文互联网上似乎很少有能直接…

Apache Log4j Server 反序列化命令执行漏洞(CVE-2017-5645)

漏洞复现环境搭建请参考 http://t.csdnimg.cn/MxmId 漏洞版本 Apache Log4j 2.8.2之前的2.x版本 漏洞验证 &#xff08;1&#xff09;开放端口4712 漏洞利用 &#xff08;1&#xff09;ysoserial工具获取 wget https://github.com/frohoff/ysoserial/releases/download/v0…

强化学习算法

从上图看出&#xff0c;强化学习可以分成价值/策略、随机策略/确定策略、在线策略/离线策略、蒙特卡洛/时间差分这四个维度。这里分析了基础算法中除了在线策略/离线策略以外的其他维度。 &#xff08;一&#xff09;基础知识 一、基础概念 重点概念&#xff1a;状态S、动作A、…

浏览器自动化~插件推荐Automa

引言 作为一款现代浏览器&#xff0c;得自动化吧&#xff0c;自主完成那些日复一日的重复性任务&#xff0c;开启音乐啥的不在话下~。而你则可以专注于其他更有意义的事情&#xff0c;如享受音乐带来的愉悦。但如果你对编写脚本一窍不通&#xff0c;又该如何实现这一愿景呢&am…

华为机考入门python3--(28)牛客28-素数伴侣

分类&#xff1a;质数、素数、贪心算法、矩阵 知识点&#xff1a; 素数里除了2&#xff0c;都是奇数 奇奇偶&#xff0c;偶&#xff0b;偶偶 对矩阵求和 sum(map(sum, matrix)) 查找元素 3 在列表中的索引 my_list.index(3) 题目来自【牛客】 质数又称素数&#xff0c;是指…

一种综合评价及决策方法:层次分析法AHP

大家好&#xff0c;层次分析法(Analytic Hierarchy Process&#xff0c;AHP)是一种多准则决策方法&#xff0c;它帮助决策者处理复杂的决策问题&#xff0c;将其分解成层次结构&#xff0c;然后通过两两比较来确定各个层次的因素之间的相对重要性。这种分析方式允许决策者对问题…

【vue与iframe通讯】

vue 与 iframe 通讯 发送数据vue 向 iframe 发送数据iframe 向 vue 发送数据接收信息( vue & iframe 通用) 实现相互通讯通讯流程图实现代码vue 页面iframe页面iframe内部重定向访问地址,更新vue路由 代码下载 前言&#xff1a;vue嵌套iframe实现步骤 发送数据 vue 向 if…