2023-6-26-第八式装饰器模式

简介: 2023-6-26-第八式装饰器模式

😉一、基础概念

装饰器模式是一种结构型设计模式,它通过动态的将责任附加到对象上来扩展对象的功能。换句话说,它提供了一种不通过子类化就能扩展对象功能的方式。装饰器模式允许向一个现有的对象添加新的功能,同时又不改变其结构。

在装饰器模式中,有一个基本的对象,称之为组件(component),它定义了一个共同的接口,可以被具体的组件或装饰器所实现。装饰器(decorator)会对组件进行包装,从而给组件增加新的功能。装饰器也实现了组件的接口,所以它们可以互相嵌套和组合,从而形成一个功能更加丰富的对象。

在代码实现上,装饰器模式使用一个装饰器类对组件进行包装,并将其传递给另一个装饰器,直到所有的装饰器都被执行完毕。这个过程中,每个装饰器都可以通过调用组件的方法来增加、修改或删除组件的行为。最终,得到的对象拥有了所有的装饰器所提供的功能。

所以,装饰器模式的优点在于:可以动态地给对象增加新的功能,而不需要修改原有的代码。缺点在于:装饰器的嵌套可能会导致复杂的代码结构,而且不正确地使用装饰器将使代码难以维护。


🐱‍🐉二、装饰器模式实现

首先,我们定义一个基础组件类Component和一个具体组件类ConcreteComponent:

class Component {
public:
    virtual void operation() = 0;
};
class ConcreteComponent : public Component {
public:
    virtual void operation() override {
        std::cout << "ConcreteComponent operation.\n";
    }
};

然后我们定义一个装饰器类Decorator,它继承自Component并持有一个指向Component对象的指针,以便在装饰过程中调用原对象的方法。Decorator还定义了一个名为addBehavior()的纯虚函数,用于在装饰过程中添加新功能。

class Decorator : public Component {
public:
    Decorator(Component* component) : component_(component) {}
    virtual void operation() override {
        if (component_)
            component_->operation();
    }
    virtual void addBehavior() = 0;
protected:
    Component* component_;
};

接下来,我们定义两个具体装饰器类ConcreteDecoratorA和ConcreteDecoratorB,它们继承自Decorator并重写了operation()和addBehavior()方法,以实现自定义的装饰功能。

class ConcreteDecoratorA : public Decorator {
public:
    ConcreteDecoratorA(Component* component) : Decorator(component) {}
    virtual void operation() override {
        Decorator::operation();
        std::cout << "ConcreteDecoratorA operation.\n";
    }
    virtual void addBehavior() override {
        std::cout << "ConcreteDecoratorA added behavior.\n";
    }
};
class ConcreteDecoratorB : public Decorator {
public:
    ConcreteDecoratorB(Component* component) : Decorator(component) {}
    virtual void operation() override {
        Decorator::operation();
        std::cout << "ConcreteDecoratorB operation.\n";
    }
    virtual void addBehavior() override {
        std::cout << "ConcreteDecoratorB added behavior.\n";
    }
};

最后,我们可以在客户端代码中使用这些类来实现装饰器模式。我们首先创建一个具体组件对象ConcreteComponent,然后将其传递给ConcreteDecoratorA对象,并调用addBehavior()方法来添加新功能。随后,我们将ConcreteDecoratorA对象传递给ConcreteDecoratorB对象再次添加新功能,并调用operation()方法,输出所有添加的功能。

int main() {
    Component* component = new ConcreteComponent();
    ConcreteDecoratorA* decoratorA = new ConcreteDecoratorA(component);
    decoratorA->addBehavior();
    ConcreteDecoratorB* decoratorB = new ConcreteDecoratorB(decoratorA);
    decoratorB->addBehavior();
    decoratorB->operation();
    delete decoratorB;
    delete decoratorA;
    delete component;
    return 0;
}

输出结果如下:

ConcreteComponent operation.
ConcreteDecoratorA operation.
ConcreteDecoratorA added behavior.
ConcreteDecoratorB operation.
ConcreteDecoratorB added behavior.

🎉三、模块之间的关系

在装饰器模式中,有四个关键角色:抽象组件、具体组件、抽象装饰器和具体装饰器。

抽象组件是定义对象的接口,可以是抽象类或接口,具体组件是实现抽象组件的类,具体装饰器是实现抽象装饰器的类,抽象装饰器是定义装饰器的接口,可以是抽象类或接口。

这四个角色之间的关系是:抽象组件可以包含具体组件,抽象装饰器可以包含抽象组件,具体装饰器可以包含抽象装饰器,具体装饰器也可以包含具体组件,形成一个以抽象装饰器为根节点的装饰器链。通过这个链,每一个装饰器都可以动态地为组件添加新的功能。

当客户端想要使用某个具体组件时,可以使用抽象装饰器来对其进行装饰,不需要知道具体组件的实现细节,这样可以实现组件与装饰器的解耦。


🐱‍🚀四、注意事项

使用桥接模式时需要注意以下事项:

  1. 抽象部分与实现部分分离。在使用桥接模式时,需要将抽象部分与实现部分分离开来,确保它们可以独立进行变化。这有助于提高系统的灵活性和可扩展性。
  2. 接口细化。在设计抽象部分和实现部分的接口时,需要尽可能细化,以确保它们的功能可以清晰地描述。
  3. 优先使用对象组合。在实现桥接模式时,需要优先使用对象组合,而不是继承。如果使用继承,可能会引发类层次结构的剧增,导致系统难以维护。
  4. 不要滥用桥接模式。桥接模式并不是一种万能模式,使用时需要根据具体情况来决定是否使用。如果使用不当,可能会导致系统的复杂度增加,反而降低系统的可维护性和可扩展性。
  5. 安全性考虑。在使用桥接模式时,需要考虑安全性的问题。特别是在桥接的两端,需要确保数据的完整性和安全性。

🎂五、使用场景

桥接模式适用于以下场景:

1.抽象和实现部分具有平行的等级结构,需要在运行时动态组合它们。桥接模式能够使抽象和实现部分各自独立变化,且能够在运行时动态组合它们,因此能够很好地适应这种场景。

2.要求不同的抽象子类可以与不同的实现子类进行组合。桥接模式能够通过抽象角色和实现角色的分离,实现不同的抽象子类可以与不同的实现子类进行组合,从而可以灵活地处理不同的组合情况。

3.一个类存在多个独立变化的维度,需要支持动态组合。桥接模式能够将各个维度的变化分离,使得每个维度的变化可以独立地扩展,从而可以非常灵活地支持多维度的动态组合。

综上所述,桥接模式适用于具有多个独立变化维度的场景,能够很好地支持这种复杂度。但需要注意的是,若使用不当,桥接模式也可能带来额外的复杂度。因此,应在设计时慎重考虑是否真正需要使用桥接模式。


🍳参考文献

🧊文章总结

提示:这里对文章进行总结:

  本文讲了关于桥接模式的知识。


目录
相关文章
|
7月前
|
JSON 监控 JavaScript
Swagger UI 本地主机教程: 如何在本地使用 Swagger UI?
Swagger UI 提供在线和离线版本,但由于各种原因,你可能需要在本地使用 Swagger UI。 在本文中,我们将向你展示如何在本地使用 Swagger UI。
|
XML JSON 前端开发
网络要素服务(WFS)详解
网络要素服务(WFS)详解
326 0
|
Linux 调度
探索操作系统核心:进程与线程管理
【10月更文挑战第24天】在数字世界的心脏,操作系统扮演着至关重要的角色。它不仅是计算机硬件与软件之间的桥梁,更是管理和调度资源的大管家。本文将深入探讨操作系统的两大基石——进程与线程,揭示它们如何协同工作以确保系统运行得井井有条。通过深入浅出的解释和直观的代码示例,我们将一起解锁操作系统的管理奥秘,理解其对计算任务高效执行的影响。
103 0
|
虚拟化 芯片
CORTEX-A7芯片中断系统基本原理与控制方法
CORTEX-A7芯片中断系统基本原理与控制方法
572 0
CORTEX-A7芯片中断系统基本原理与控制方法
|
弹性计算 监控 NoSQL
【Elastic Engineering】Beats:Beats 入门教程 (二)
这篇文章是 “Beats 入门教程 (一)”的续篇。在上一篇文章,我们主要讲述了 Beats 的一些理论方面的知识。在这篇文章中,我们将具体展示如何使用 Filebeat 及 Metriceat 把数据导入到我们的 Elasticsearch 并对他们进行分析。
443 0
【Elastic Engineering】Beats:Beats 入门教程 (二)
|
3天前
|
弹性计算 运维 搜索推荐
三翼鸟携手阿里云ECS g9i:智慧家庭场景的效能革命与未来生活新范式
三翼鸟是海尔智家旗下全球首个智慧家庭场景品牌,致力于提供覆盖衣、食、住、娱的一站式全场景解决方案。截至2025年,服务近1亿家庭,连接设备超5000万台。面对高并发、低延迟与稳定性挑战,全面升级为阿里云ECS g9i实例,实现连接能力提升40%、故障率下降90%、响应速度提升至120ms以内,成本降低20%,推动智慧家庭体验全面跃迁。
|
4天前
|
数据采集 人工智能 自然语言处理
3分钟采集134篇AI文章!深度解析如何通过云无影AgentBay实现25倍并发 + LlamaIndex智能推荐
结合阿里云无影 AgentBay 云端并发采集与 LlamaIndex 智能分析,3分钟高效抓取134篇 AI Agent 文章,实现 AI 推荐、智能问答与知识沉淀,打造从数据获取到价值提炼的完整闭环。
365 91
|
4天前
|
SQL 人工智能 自然语言处理
Geo优化SOP标准化:于磊老师的“人性化Geo”体系如何助力企业获客提效46%
随着生成式AI的普及,Geo优化(Generative Engine Optimization)已成为企业获客的新战场。然而,缺乏标准化流程(Geo优化sop)导致优化效果参差不齐。本文将深入探讨Geo专家于磊老师提出的“人性化Geo”优化体系,并展示Geo优化sop标准化如何帮助企业实现获客效率提升46%的惊人效果,为企业在AI时代构建稳定的流量护城河。
378 156
Geo优化SOP标准化:于磊老师的“人性化Geo”体系如何助力企业获客提效46%