装饰器模式【结构型模式C++】

1.概述

装饰器模式是一种结构型设计模式, 允许你通过将对象放入包含行为的特殊封装对象中来为原对象绑定新的行为。

2.结构

  • 抽象构件(Component)角色:定义一个抽象接口以规范准备接收附加责任的对象。
  • 具体构件(Concrete Component)角色:实现抽象构件,通过装饰角色为其添加一些职责。
  • 抽象装饰(Base Decorator)角色:继承抽象构件,并包含具体构件的实例,可以通过其子类扩展具体构件的功能。
  • 具体装饰(ConcreteDecorator)角色:实现抽象装饰的相关方法,并给具体构件对象添加附加的责任。

3.实现

3.1 实例类比

以对新买的手机为例,当买了一个手机,一般都会贴保护膜,加装手机壳,有的还会加装各类挂失。下面是实现此过程的类图:

3.2 具体实现
cpp 复制代码
#include <iostream>
#include <string>
#include<vector>

using namespace std;

//手机抽象类	
class Phone
{
public:
	virtual void Do() = 0;
};
	

//实现抽象接口
class Huawei:public Phone
{
	 void Do()override{
		 cout << "对华为手机开始装饰"<< endl;
	}
};
	
// 装饰抽象角色------继承抽象构件角色,包含一个抽象构件角色对象的实例	
class Decorator :public Phone
{
protected:
	Phone *phone_;

public:
	Decorator(Phone* phone):phone_(phone){
	}
	void Do(){
		if (phone_ != nullptr){
			phone_->Do();
		}
	}
};
	
//具体装饰角色------手机贴上保护膜
	
class Protectorfilm :public Decorator
{
public:
	Protectorfilm(Phone* phone) : Decorator(phone) {
	}

	 void Do(){
		    phone_->Do();
			Addfilm();
	}

	void Addfilm(){
		cout << "手机贴保护膜" << endl;
	}
};

//具体装饰角色------手机保护壳
class Shell :public Decorator
{
public:
		Shell(Phone* phone) : Decorator(phone) {
	}
	
	void Do(){
		phone_->Do();
		AddShell();
	}

	void AddShell(){
		cout << "手机安装保护壳" << endl;
	}
};

//具体装饰角色------手机挂饰	
class Accessory :public Decorator
{
public:
	Accessory(Phone* phone) : Decorator(phone) { 
	}
    
	void Do(){
		phone_->Do();
		AddAccessory();
	}

    void AddAccessory(){
		cout << "手机安装挂件" << endl;		
	}
};




int main()
{
	cout << "===装饰模式===" << endl;
	//现在获得了一个手机
	Phone *phone = new Huawei();

	//裸机一个,先贴保护膜
	Decorator*protectorfilm = new Protectorfilm(phone);
	protectorfilm->Do();
	cout << "=======================" << endl;

	//再加个保护外壳
	Decorator *shell = new Shell(protectorfilm);
	shell->Do();
	cout << "=======================" << endl;

	//最后加装挂件
	Decorator *accessory = new Accessory(shell);
	accessory->Do();
	cout << "=======================" << endl;
}

3.3运行结果 

4.装饰器模式优缺点

优点:

  1. 装饰模式和继承关系的目的都是拓展对象的功能,但是装饰模式可以提供比继承更多的灵活性。装饰模式允许系统动态的决定"贴上"一个需要的"装饰",或者出掉一个不需要的"装饰",继承关系不同,继承是静态的,它在系统运行前就决定了。
  2. 通过使用不同具体的装饰类以及这些装饰类的组合,可以创造出很多不同行为的组合。
  3. 比继承更多的灵活性,每一种不同的组合均需要事先通过子类的继承方式设计好。

缺点:

  1. 会产生比继承关系更多的对象。使得查错变得困难,特别是这些对象看上去都很相像。

5 应用场景

  • 用于扩展一个类的功能或者给一个类添加附加职责
  • 不能通过继承的方式对现有系统的类进行功能扩展时,可以考虑使用装饰模式进行功能扩展。
  • 不能使用继承方式的原因是系统现有大量独立的类,如果通过继承方式进行扩展,可能会出现类爆炸;
相关推荐
再__努力1点25 分钟前
【77】积分图像:快速计算矩形区域和核心逻辑
开发语言·图像处理·人工智能·python·算法·计算机视觉
Evand J33 分钟前
【2026课题推荐】基于小波/互相关/FFT的卡尔曼滤波的轨迹估计,及MATLAB例程的运行结果
开发语言·matlab·目标跟踪·轨迹跟踪
独自归家的兔40 分钟前
Java Robot 详解:系统级鼠标 / 键盘模拟的核心原理与实战
java·开发语言
小灰灰搞电子1 小时前
Qt 开发环境选择Qt Creator、Visual Studio还是 VS Code?
开发语言·qt·visual studio
何中应1 小时前
Bean的三种注入方式
开发语言·spring boot·后端·spring
胖咕噜的稞达鸭1 小时前
算法日记专题:位运算II( 只出现一次的数字I II III 面试题:消失的两个数字 比特位计数)
c++·算法·动态规划
茉莉玫瑰花茶1 小时前
ProtoBuf - 3
服务器·c++·protobuf
wanghowie1 小时前
01.03 Java基础篇|面向对象核心与设计实践
java·开发语言
Algebraaaaa1 小时前
为什么线程阻塞要用.join而不是.wait
java·c++·python
巴拉巴拉~~1 小时前
Flutter 通用滑块组件 CommonSliderWidget:单值 / 范围 + 刻度 + 标签 + 样式自定义
开发语言·前端·javascript