Decorator

news2025/1/16 9:10:49

Decorator

动机
  • 在某些情况下我们可能会“过度地使用继承来扩展对象的功能”, 由于继承为类型引入的静态特质,使得这种扩展方式缺乏灵活性; 并且随着子类的增多(扩展功能的增多),各种子类的组合(扩展 功能的组合)会导致更多子类的膨胀
  • 如何使”对象功能的扩展“能够根据需要动态实现,同时避免”扩展功能的增多“带来的子类膨胀问题,从而使得任何”功能扩展变化“所导致的影响降到最低?
模式定义

动态(组合)地给一个对象增加一些额外的职责。就增加功能而言,Decorator模式比生成子类(继承)更为灵活(消除重复代码 & 减少子类个数)。

在这里插入图片描述

#include <iostream>

class Component
{
public:
    virtual ~Component() {}

    virtual void operation() = 0;
    // ...
};

class ConcreteComponent : public Component
{
public:
    ~ConcreteComponent() {}

    void operation()
    {
        std::cout << "Concrete Component operation" << std::endl;
    }
    // ...
};

class Decorator : public Component // is-a 为了完善接口规范
{
private:
    Component *component; // has-a 为了将来具体的实现

public:
    ~Decorator() {}

    Decorator(Component *c) : component(c) {}

    virtual void operation()
    {
        component->operation();
    }
    // ...
};

class ConcreteDecoratorA : public Decorator
{
public:
    ConcreteDecoratorA(Component *c) : Decorator(c) {}

    void operation()
    {
        Decorator::operation();
        std::cout << "Decorator A" << std::endl;
    }
    // ...
};

class ConcreteDecoratorB : public Decorator
{
public:
    ConcreteDecoratorB(Component *c) : Decorator(c) {}

    void operation()
    {
        Decorator::operation();
        std::cout << "Decorator B" << std::endl;
    }
    // ...
};

int main()
{
    ConcreteComponent *cc = new ConcreteComponent();
    ConcreteDecoratorB *db = new ConcreteDecoratorB(cc);
    ConcreteDecoratorA *da = new ConcreteDecoratorA(db);

    Component *component = da;
    component->operation(); 

    delete da;
    delete db;
    delete cc;

    return 0;
}
/*
Concrete Component operation
Decorator B
Decorator A
*/
何时场景
  • 动态且透明地向各个对象添加职责,即不影响其他对象
  • 对于可以撤销的责任
  • 当通过子类化扩展不切实际时
总结
  • 通过采用组合而非继承的手法, Decorator模式实现了在运行时动态扩展对象功能的能力,而且可以根据需要扩展多个功能。避免 了使用继承带来的“灵活性差”和“多子类衍生问题
  • Decorator类在接口上表现为is-a Component的继承关系,即Decorator类继承了Component类所具有的接口(为了完善接口规范)。但在实现上又表现为has-a Component的组合关系,即Decorator类又使用了另外一个Component类(为了将来的具体实现)
  • Decorator模式的目的并非解决“多子类衍生的多继承”问题, Decorator模式应用的要点在于解决“主体类在多个方向上的扩展功能”——是为“装饰”的含义。

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

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

相关文章

Ubuntu右上角不显示网络的图标解决办法

一.line5改为true sudo vim /etc/NetworkManager/NetworkManager.conf 二.重启网卡 sudo service network-manager stop sudo mv /var/lib/NetworkManager/NetworkManager.state /tmp sudo service network-manager start

全网最详细的本地搭建GitLab代码仓库教学

大体的步骤 本地安装VMware虚拟机。然后再虚拟机上安装CentOs7镜像系统。在Linux中安装GitLab-Ce。在Linux中安装GitLab-Runner在自己搭建的GitLab上面创建一个项目&#xff0c;然后拉取到本地然后提交之后实现自动化部署。 步骤一 这一步骤我在之前我的安装Redis文章中有讲解…

关于Jupyter notebook 创建python3 时进去不能重命名问题及不能编程问题

首先写这篇博客时&#xff0c;已经被这个问题折磨了三天&#xff0c;看了很多博客&#xff0c;其实解决这个问题的关键就是要么没有下pyzmq或者等级太高&#xff0c;要么等级太低&#xff0c;首先我会按照我思路来。 问题如图&#xff1a; 1.自动换行 2.不能重命名 我的解决办…

学习记忆——数学篇——案例——算术——绝对值三角不等式

第一步&#xff1a;记住公式&#xff0c;绝对值差&#xff0c;和差绝对值&#xff0c;绝对值和 第二步&#xff1a;记住口诀&#xff1a;取等条件&#xff1a;中间相加取等号&#xff0c;左异右同零取到&#xff1b;中间相减取等号&#xff0c;上面符号方向调 题型记忆法 歌诀记…

【Cents OS7 安装 Docker以及DockerCompose】

文章目录 0.安装Docker1.CentOS安装Docker1.1.卸载&#xff08;可选&#xff09;1.2.安装docker1.3.启动docker1.4.配置镜像加速 2.CentOS7安装DockerCompose2.1.下载2.2.修改文件权限2.3.Base自动补全命令&#xff1a; 3.Docker镜像仓库3.1.简化版镜像仓库3.2.带有图形化界面版…

SLAM从入门到精通(ROS和底盘Stm32的关系)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 学过Ros的同学&#xff0c;一般对subscribe、publish、话题、服务这些内容都比较熟悉。如果再熟悉一点的话&#xff0c;还会知道slam、move_base、…

好物周刊#10:算法学习必备

https://cunyu1943.github.io https://yuque.com/cunyu1943 村雨遥的好物周刊&#xff0c;记录每周看到的有价值的信息&#xff0c;主要针对计算机领域&#xff0c;每周五发布。 一、项目 1. ChatGPT-Midjourney 该项目基于 ChatGPT-Next-Web 开发&#xff0c;只要拥有自己…

聪明应对工程项目管理难题的方法和技巧

对于国内的工程项目管理中&#xff0c;经常需要面对以下几个问题&#xff1a; 1.项目问题相互牵扯&#xff0c;积累成堆&#xff1a;通常一个问题不能及时的解决&#xff0c;就会导致更多的任务无法完成&#xff0c;问题越堆积越严重。 2.项目延期&#xff0c;增加成本&#xf…

【leetcode】 vscode leetcode [ERROR] invalid password? 问题解决

目录 问题解决 问题 使用vscode连接leetcode出现下列问题&#xff1a; vscode leetcode [ERROR] invalid password?出现invalid password?的问题&#xff0c;首先需要检查账号密码是否出错&#xff0c;leetcode的账号可以是手机或邮箱&#xff0c;然后密码去check一下&…

Anaconda3-2023.07-2安装和配置教程

Anaconda3-2023.07-2安装和配置教程 前言一、彻底卸载python1、卸载python2、删除环境变量 二、下载Anaconda方式一&#xff1a;官网下载方式二、镜像站下载 三、安装Anaconda3四、配置环境变量五、检验安装是否成功1、查看anaconda版本2、查看python版本3、查看Anaconda Navif…

C (1094) : DS双向链表—前驱后继

Description 在双向链表中&#xff0c;A有一个指针指向了后继节点B&#xff0c;同时&#xff0c;B又有一个指向前驱节点A的指针。这样不仅能从链表头节点的位置遍历整个链表所有节点&#xff0c;也能从链表尾节点开始遍历所有节点。 对于给定的一列数据&#xff0c;按照给定的…

springboot2.7.15集成springcloud,springcloudalibaba

spring-cloud官方中文文档 spring-cloud官方文档 Spring Cloud Alibaba官方文档 这边用的是springboot2.7.15以下是集成springcloud,springcloudalibaba的版本推荐 <dependencyManagement><dependencies><!--SpringBoot整合Spring Cloud--><dependency…

梯度裁剪:torch.nn.utils.clip_grad_norm_详解

梯度裁剪是为了防止梯度爆炸。在训练FCOS算法时&#xff0c;因为训练过程出现了损失为NaN的情况&#xff0c;在github issue有很多都是这种训练过程出现loss为NaN&#xff0c;使用torch.nn.utils.clip_grad_norm_梯度裁剪函数&#xff0c;可以有效预防梯度爆炸的情况发生。 1 …

OpenCV4(C++)—— 图像阈值化

文章目录 前言一、固定阈值化 —— threshold二、自适应阈值化 —— adaptiveThreshold三、LUT查找表 前言 图像阈值化在许多计算机视觉和图像处理任务中都是一个重要的预处理步骤。在边缘检测过程中&#xff0c;通过将图像转换为二值图像&#xff0c;可以突出图像中的边缘信息…

Qt:多语言支持,构建全面应用程序“

Qt&#xff1a;强大API、简化框架、多语言支持&#xff0c;构建全面应用程序" 强大的API&#xff1a;Qt提供了丰富的API&#xff0c;包括250多个C类&#xff0c;基于模板的集合、序列化、文件操作、IO设备、目录管理、日期/时间等功能。还包括正则表达式处理和支持2D/3D…

制作长图海报的详细指南,制作长图海报的5个步骤

制作长图海报是宣传活动、产品或服务的重要方式之一。乔拓云后台提供了丰富的海报模板&#xff0c;让你轻松制作出专业级的长图海报。下面将介绍如何使用乔拓云后台制作长图海报的技巧。 一、选择模板 首先&#xff0c;注册并登录乔拓云后台&#xff0c;进入云设计页面。在选择…

C语言学生成绩录入系统

一、系统概述 该系统是一个由链表创建主菜单的框架&#xff0c;旨在快速创建学生成绩录入系统的主菜单结构。其主要任务包括&#xff1a; 实现链表的创建、插入和遍历功能&#xff0c;用于存储和展示学生成绩录入系统各个模块的菜单项。 2. 提供用户友好的主菜单界面&#xf…

证件照换底色详细教程

说到证件照的底色更改&#xff0c;我想对大部分朋友来说是蛮头疼的事情&#xff0c;由于我们不论是在生活还是学习中&#xff0c;有时候总会要上传一些证件照&#xff0c;而当你手上有证件照准备上传时&#xff0c;发现底色不对&#xff0c;是不是很抓狂&#xff0c;现在&#…

SpringCloud学习笔记-Eureka的服务拉取

假设是OrderService里面拉取Eureka的服务之一User Service 1.依然需要在该服务里面引入依赖 <dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-netflix-eureka-client</artifactId> </dependenc…

水波纹文字效果动画

效果展示 CSS 知识点 text-shadow 属性绘制立体文字clip-path 属性来绘制水波纹 工具网站 CSS clip-path maker 效果编辑器 页面整体结构实现 使用多个 H2 标签来实现水波纹的效果实现&#xff0c;然后使用clip-path结合动画属性一起来进行波浪的起伏动画实现。 <div …