c++设计模式之装饰器模式

作用

为现有类增加功能

案例说明

cpp 复制代码
class Car
{
public:
    virtual void show()=0;    
};

class Bmw:public Car
{
public:
    void show(){cout<<"宝马汽车>>"<<endl;}
};

class Audi:public Car
{
public:
    void show(){cout<<"奥迪汽车>>"<<endl;}
};

class Bcw:public Car{
public:
    void show(){cout<<"奔驰汽车>>"<<endl;}
};

如上,我们定义了汽车抽象类,并定义三个具体的汽车类。

现在的问题是:如果需要给这三种汽车都增加仨个功能------定速导航、刹车、车道偏离。应该如何实现?

  • 第一种办法,为每个汽车类都定义两个功能子类,如下图所示
  • 第二种办法,使用装饰器,只需要给Car下增加一个功能类
cpp 复制代码
class CarFunction:public Car{
public:
    CarFunction(Car* carPtr):_carPtr(carPtr){};
protected :
    Car* _carPtr;
};
//装饰器1,定速导航
class ConcreteDecorator:public CarFunction
{
public:
    ConcreteDecorator(Car* carPtr):CarFunction(carPtr){};
    void show()
    {
        _carPtr->show();
        cout<<"\t定速巡航"<<endl;
    }
};

//装饰器2,刹车
class BrakeDecorator:public CarFunction
{
public:
    BrakeDecorator(Car* carPtr):CarFunction(carPtr){}
    void show()
    {
        _carPtr->show();
        cout<<"\t刹车"<<endl;
    }
};

//车道偏离
class LaneDecorator:public CarFunction
{
public:
    LaneDecorator(Car* carPtr):CarFunction(carPtr){}
    void show(){
        _carPtr->show();
        cout<<"\t车道偏离"<<endl;
    }
};

代码测试

cpp 复制代码
void test()
{
    Car* p1=new ConcreteDecorator(new Bmw());//使用宝马汽车定位导航功能
    Car* p2=new BrakeDecorator(new Audi());//使用奥迪汽车请假刹车功能
    Car* p3=new LaneDecorator(new Bcw());//使用奔驰汽车车道偏离功能
    p1->show();
    p2->show();
    p3->show();
}

可以看到,使用装饰器模式给类增加功能的优点就是避免了大量的代码重复,假如使用继承子类的方式给各种汽车类增加功能,这种重复的功能增加会产生大量冗余的代码

对比代理模式(详见c++设计模式之代理模式-CSDN博客),装饰器模式和代理模式的设计都有许多相似之处

他们都通过将真实的对象包装到代理类和装饰类中,或者说把实际的操作对象放到代理类或装饰类中来达成目的 。他们利用的特性都是基于c++的多态特性

但他们的目的不同

  • 代理模式 主要是为了控制对真实对象的访问
  • 装饰器模式 主要是为了给真实对象增加功能
相关推荐
智者知已应修善业34 分钟前
【51单片机8位数码管动态显示日期小数点风格】2023-11-13
c++·经验分享·笔记·算法·51单片机
智者知已应修善业34 分钟前
【51单片机有三个LED 分别第一个灯闪三下 再到第二个灯又闪三下 再到第三个灯又闪三下 就这样循环程序】2023-11-16
c++·经验分享·笔记·算法·51单片机
Doris_20233 小时前
代码格式化 使用oxfmt
设计模式·架构·前端框架
Doris_20233 小时前
说一说ESLint+Prettier生效的原理
前端·设计模式·架构
Pomelooooo4 小时前
把 git commit 这件事,彻底交给 AI ——一个工程化 /git-commit 命令的设计与落地
设计模式
玖釉-5 小时前
二叉树展开为链表:从先序遍历到原地指针重排
c++·windows·算法·leetcode·链表
Mister西泽5 小时前
C++ Primer Plus 第六版 编程练习题及详细答案
开发语言·c++·学习·visual studio
Qt程序员5 小时前
从上电到系统就绪:ARM+U-Boot 嵌入式 Linux 启动流程
linux·运维·c++·内核·设备树·嵌入式·ram
invicinble6 小时前
设计模式(类的拓扑结构)(描述总纲)
设计模式·原型模式
cany10006 小时前
C++ -- lambda捕获
c++