裝飾模式用于動态的往元件(Component)上添加功能。主要使用的場所為:
- 需要擴充一個類的功能,或給一個類增加附加責任。
- 需要動态的給一個對象增加功能,這些功能可以再動态地撤銷。
- 需要增加一些基本功能的排列組合而産生的非常大量的功能。
就上圖而言,待擴充的類就是ConcreteComponent,附屬的職責或者功能就是ConcreteDecorator身上帶有的工能。裝飾模式比較強大的地方在于:1)可以動态添加或者撤銷職責或者功能。2)可以實作功能的排列組合而不用通過繼承的方式。
舉一個穿衣服的例子,UML圖如上,代碼如下:
#include <string>
#include <unordered_set>
#include <iostream>
using namespace std;
class CAbstractAvator {
public:
virtual void Decorate() = ;
};
class CConcreteAvator : public CAbstractAvator{
string m_name;
public:
CConcreteAvator(const string& name) : m_name(name) {}
void Decorate() override
{
cout << "I'm " << m_name << ", I'm wearing: " << endl;
}
};
class CAbstractDecorator : public CAbstractAvator{
protected:
CAbstractAvator* pAvator = nullptr;
public:
CAbstractDecorator(CAbstractAvator* p) : pAvator(p) {}
};
class CUnderWearDecorator : public CAbstractDecorator {
public:
CUnderWearDecorator(CAbstractAvator* p) : CAbstractDecorator(p) {}
void Decorate() override {
pAvator->Decorate();
cout << "Underwear" << endl;
}
};
class CTshirtDecorator : public CAbstractDecorator {
public:
CTshirtDecorator(CAbstractAvator* p) : CAbstractDecorator(p) {}
void Decorate() override {
pAvator->Decorate();
cout << "Tshirt" << endl;
}
};
class CSweatersDecorator : public CAbstractDecorator {
public:
CSweatersDecorator(CAbstractAvator* p) : CAbstractDecorator(p) {}
void Decorate() override {
pAvator->Decorate();
cout << "Sweaters" << endl;
}
};
class CCoatDecorator : public CAbstractDecorator {
public:
CCoatDecorator(CAbstractAvator* p) : CAbstractDecorator(p) {}
void Decorate() override {
pAvator->Decorate();
cout << "Coat" << endl;
}
};
void main()
{
CAbstractAvator *pAvator = new CConcreteAvator("avator");
pAvator = new CUnderWearDecorator(pAvator);
pAvator = new CTshirtDecorator(pAvator);
pAvator = new CSweatersDecorator(pAvator);
pAvator = new CCoatDecorator(pAvator);
pAvator->Decorate();
}
輸出結果如下:
I'm avator, I'm wearing:
Underwear
Tshirt
Sweaters
Coat