天天看点

C++实现设计模式——装饰者(decorator)模式C++实现设计模式——装饰者(decorator)模式

C++实现设计模式——装饰者(decorator)模式

  • 装饰者(decorator)模式定义

  动态地给一个对象添加一些额外的职责。就增加功能来说,Decorator模式相比生成子类更为灵活。不改变接口的前提下,增强所考虑的类的性能。

  • 何时使用

    1)需要扩展一个类的功能,或给一个类增加附加责任。

    2)需要动态的给一个对象增加功能,这些功能可以再动态地撤销。

    3)需要增加一些基本功能的排列组合而产生的非常大量的功能。

注意:装饰者模式的根本是装饰而不是改变接口,原先调用的什么接口,最后还是会调用到这个接口,也就是装饰角色的接口与抽象构件角色的接口是完全一致的,也叫做完全透明装饰者模式,反之如果接口发生了改变,那就是一种适配器了。

举个通俗的例子:

我们大家都玩游戏吧,都知道有皮肤这个东西,假设原始角色展示是通过一个show这个接口来展现游戏角色的,皮肤就是一个装饰器,那穿戴了皮肤就是调用了装饰器的show接口,区别在于装饰器的show里面比普通的原始的show接口多了添加皮肤的逻辑,装饰器show在添加了获取皮肤的逻辑后,最后还是会调用原来的原始show接口进行展示。它的根本还是用原始的show展示角色,没有改变原来的接口,综上就是兜兜转转还是你的那种感觉。

再给大家举一个实际中会用到的例子,大家在开发中经常会用到生成日志文件吧,而往往不同的场合会生成不同格式的日志内容,比如有json的,xml的,表格的各种形式,如果用继承的方式就需要在打印日志的类下不停的派生各种子类,然后重载日志输出接口,这样类的复杂度越来越高。此时如果用装饰者,可以需要什么模式的时候就装饰什么。代码如下:

#include<iostream>
using namespace std;

class log {          //整个日志类
    public:
        virtual void log_out(string filename, string  value) {
            cout<<"输出日志"<<value<<"到"<<filename<<"文件"<<endl;
        }
};

class errlog:public log {    //错误日志输出类
    public:
    void log_out(string filename, string  value) {
            cout<<"输出错误日志"<<value<<"到"<<filename<<"文件"<<endl;
        }
};

class debuglog:public log {  //debug日志输出类
    public:
    void log_out(string filename, string  value) {
            cout<<"输出debug日志"<<value<<"到"<<filename<<"文件"<<endl;
        }
}; 

class decoratorlog:public log {//日志输出装饰器
    public:
        log *m_logout;
        decoratorlog() {}
        decoratorlog(log *logout) {
            this->m_logout = logout;
        }
        void log_out(string filename, string  value) {
                cout<<"输出debug日志"<<value<<"到"<<filename<<"文件"<<endl;
        }
};

class jsondecorator:public decoratorlog {//json日志输出装饰器
    public:
        jsondecorator(log *logout) {
            this->m_logout = logout;
        }
        void log_out(string filename, string  value) {
            cout<<"value进行了json转换"<<endl;     //装饰器的作用,也就是需要扩充的新功能需求
            m_logout->log_out(filename,value);    //调用到原先的接口
        }
};

class xmldecorator:public decoratorlog {//json日志输出装饰器
    public:
        xmldecorator(log *logout) {
            this->m_logout = logout;
        }
        void log_out(string filename, string  value) {
            cout<<"value进行了xml转换"<<endl;      //装饰器的作用,也就是需要扩充的新功能需求
            m_logout->log_out(filename,value);    //调用到原先的接口
        }
};
int main() {
    log *log1 = new jsondecorator( new debuglog());
    log1->log_out("1.json","hello");
    log *log2 = new xmldecorator( new errlog());
    log2->log_out("1.xml","world");
    return 0;
}

/*
value进行了json转换
输出debug日志hello到1.json文件
value进行了xml转换
输出错误日志world到1.xml文件
*/
           
C++实现设计模式——装饰者(decorator)模式C++实现设计模式——装饰者(decorator)模式

这个模式的思想就是,装饰者就只是装饰,无论用到哪个装饰器子类,最后还是会调用到实际的log_out接口函数。

  • 装饰者模式的结构
C++实现设计模式——装饰者(decorator)模式C++实现设计模式——装饰者(decorator)模式

继续阅读