Flyweight享元模式
作用:运用共享技术有效地支持大量细粒度的对象。
内部状态intrinsic和外部状态extrinsic:
1)Flyweight模式中,最重要的是将对象分解成intrinsic和extrinsic两部分。
2)内部状态:在享元对象内部并且不会随环境改变而改变的共享部分,可以称为是享元对象的内部状态
3)外部状态:而随环境改变而改变的,取决于应用环境,或是实时数据,这些不可以共享的东西就是外部状态了。
4)内部状态和外部状态之间的区别:
在Flyweight模式应用中,通常修改的是外部状态属性,而内部状态属性一般都是用于参考或计算时引用。
Flyweight执行时所需的状态必定是内部的或外部的。内部状态存储于ConcreteFlyweight对象之中;而外部状态则由Client对象存储或计算。当用户调用Flyweight对象的操作时,将该状态传递给它。
以文字处理软件为例:
内部状态存储于flyweight中,它包含了独立于flyweight场景的信息,这些信息使得flyweight可以被共享。如字符代码,字符大小……
外部状态取决于flyweight场景,并根据场景而变化,因此不可共享。用户对象负责在必要的时候将外部状态传递给flyweight,如字符位置,字符颜色……
UML图:
解析:
Flyweight:享元类的基类,定义一个接口,通过这个接口Flyweight可以接受并作用于外部状态。
ConcreteFlyweight:实现Flyweight接口, 并为内部状态( 如果有的话) 增加存储空间。ConcreteFlyweight对象必须是可共享的。它所存储的状态必须是内部的(intrinsic);即,它必须独立于ConcreteFlyweight对象的场景。
UnsharedConcreteFlyweight:并非所有的Flyweight子类都需要被共享。Flyweight接口使共享成为可能,但它并不强制共享。在Flyweight对象结构的某些层次,UnsharedConcreteFlyweight对象通常将ConcreteFlyweight对象作为子节点。
FlyweightFactory:
1) 创建并管理Flyweight对象。
2)确保合理地共享Flyweight。当用户请求一个Flyweight时,FlyweightFactory对象提供一个已创建的实例或者创建一个(如果不存在的话)
Client
1)维持一个对Flyweight的引用。
2)计算或存储一个(多个)Flyweight的外部状态。
分析:
享元模式可以避免大量非常相似类的开销。在程序设计中,有时需要生成大量细粒度的类实例来表示数据。如果能发现这些实例数据除了几个参数外基本都是相同的。有时就能够大幅度地减少实例化的类的数量。如果能把那些参数移到类实例的外面,在方法调用时将它们传递进来,就可以通过共享大幅度地减少单个实例的数目。
比如在文档编辑器的设计过程中,我们如果为没有字母创建一个对象的话,系统可能会因为大量的对象而造成存储开销的浪费。例如一个字母“a”在文档中出现了100000次,而实际上我们可以让这一万个字母“a”共享一个对象,当然因为在不同的位置可能字母“a”有不同的显示效果(例如字体和大小等设置不同),在这种情况我们可以为将对象的状态分为“外部状态”和“内部状态”,将可以被共享(不会变化)的状态作为内部状态存储在对象中,而外部对象(例如上面提到的字体、大小等)我们可以在适当的时候将外部对象最为参数传递给对象(例如在显示的时候,将字体、大小等信息传递给对象)。
Flyweight的内部状态是用来共享的,Flyweightfactory负责维护一个Flyweight池(存放内部状态的对象),当客户端请求一个共享Flyweight时,这个factory首先搜索池中是否已经有可适用的,如果有,factory只是简单返回送出这个对象,否则,创建一个新的对象,加入到池中,再返回送出这个对象.池为重复或可共享的对象、属性设置一个缓冲,称为内部状态。这些内部状态一般情况下都是不可修改的,也就是在第一个对象、属性被创建后,就不会去修改了(否则就没意义了)。
Flyweight 对对象的内部状态进行共享,只为每种内部状态创建一个实例,对内部状态使用了单例模式。
用户不应直接对ConcreteFlyweight类进行实例化,而只能从FlyweightFactory对象得到ConcreteFlyweight对象,这可以保证对它们适当地进行共享。
存储节约由以下几个因素决定:
1) 因为共享,实例总数减少的数目
2) 对象内部状态的平均数目
3) 外部状态是计算的还是存储的
实现要点
1)享元工厂维护一张享元实例表。
2)享元不可共享的状态需要在外部维护。即删除外部状态:该模式的可用性在很大程度上取决于是否容易识别外部状态并将它从共享对象中删除。
3)按照需求可以对享元角色进行抽象。
4)管理共享对象:引用计数和垃圾回收……
何时采用
1)如果一个应用程序使用了大量的对象,而大量的这些对象造成了很大的存储开销时就应该考虑使用;
2)还有就是对象的大多数状态可变为外部状态,如果删除对象的外部状态,那么可以用相对较少的共享对象取代很多组对象,此时可以考虑所使用享元模式。
3)系统中有大量耗费了大量内存的细粒度对象,并且对外界来说这些对没有任何差别的(或者说经过改造后可以是没有差别的)。
在文档编辑器例子中如果一个字符对应一个对象,那么一篇文档所要容纳的对象将是非常的庞大耗费大量的内存。而实际组成文档的字符是有限的,是由这些字符不同的组合和排列得到的。所以需要共享,将基本的字符进行共享,将使得字符对象变得有限。
代码如下:
Flyweight.h
1 #ifndef _FLYWEIGHT_H_ 2 #define _FLYWEIGHT_H_ 3 4 #include <string> 5 #include <vector> 6 7 using namespace std; 8 9 //基类,定义操作接口Operation 10 class Flyweight 11 { 12 public: 13 //操作外部状态extrinsicState 14 virtual void Operation(const string& extrinsicState)=0; 15 string GetIntrinsicState(); 16 virtual ~Flyweight(); 17 protected: 18 Flyweight(string intrinsicState); 19 private: 20 //内部状态,也可以放在ConcreteFlyweight中 21 string _intrinsicState; 22 }; 23 24 class ConcreteFlyweight:public Flyweight 25 { 26 public: 27 //实现接口函数 28 virtual void Operation(const string& extrinsicState); 29 ConcreteFlyweight(string intrinsicState); 30 ~ConcreteFlyweight(); 31 }; 32 33 class UnsharedConcreteFlyweight:public Flyweight 34 { 35 public: 36 virtual void Operation(const string& extrinsicState); 37 UnsharedConcreteFlyweight(string intrinsicState); 38 ~UnsharedConcreteFlyweight(); 39 }; 40 41 class FlyweightFactory 42 { 43 public: 44 FlyweightFactory(); 45 ~FlyweightFactory(); 46 //获得一个请求的Flyweight对象 47 Flyweight* GetFlyweight(string key); 48 //获取容器中存储的对象数量 49 void GetFlyweightCount(); 50 protected: 51 private: 52 //保存内部状态对象的容器 53 vector<Flyweight*> m_vecFly; 54 }; 55 #endif
Flyweight.cpp
1 #include "Flyweight.h" 2 #include <iostream> 3 4 using namespace std; 5 6 Flyweight::Flyweight(string intrinsicState) 7 { 8 this->_intrinsicState = intrinsicState; 9 } 10 11 Flyweight::~Flyweight() 12 {} 13 14 string Flyweight::GetIntrinsicState() 15 { 16 return this->_intrinsicState; 17 } 18 19 ConcreteFlyweight::ConcreteFlyweight(string intrinsicState):Flyweight(intrinsicState) 20 { 21 } 22 23 ConcreteFlyweight::~ConcreteFlyweight() 24 { 25 } 26 27 void ConcreteFlyweight::Operation(const string& extrinsicState) 28 { 29 cout << this->GetIntrinsicState() << endl; 30 cout << extrinsicState << endl; 31 } 32 33 UnsharedConcreteFlyweight::UnsharedConcreteFlyweight(string intrinsicState):Flyweight(intrinsicState) 34 { 35 } 36 37 UnsharedConcreteFlyweight::~UnsharedConcreteFlyweight() 38 { 39 } 40 41 void UnsharedConcreteFlyweight::Operation(const string& extrinsicState) 42 { 43 cout << "extrinsicState" << endl; 44 } 45 46 FlyweightFactory::FlyweightFactory() 47 {} 48 49 FlyweightFactory::~FlyweightFactory() 50 {} 51 52 //若该对象已存在,直接返回,否则新建一个对象,存入容器中,再返回 53 Flyweight* FlyweightFactory::GetFlyweight(string key) 54 { 55 vector<Flyweight*>::iterator iter = this->m_vecFly.begin(); 56 for(;iter!= this->m_vecFly.end();iter++) 57 { 58 if((*iter)->GetIntrinsicState() == key) 59 { 60 return *iter; 61 } 62 } 63 Flyweight* fly = new ConcreteFlyweight(key); 64 this->m_vecFly.push_back(fly); 65 return fly; 66 } 67 68 void FlyweightFactory::GetFlyweightCount() 69 { 70 cout << this->m_vecFly.size() << endl; 71 }
main.cpp
1 #include "Flyweight.h" 2 #include <iostream> 3 #include <string> 4 5 using namespace std; 6 7 int main() 8 { 9 //外部状态extrinsicState 10 string extrinsicState = "ext"; 11 12 //工厂对象,工厂对象 13 FlyweightFactory* fc = new FlyweightFactory(); 14 15 //向工厂申请一个Flyweight对象,且该对象的内部状态值为“hello” 16 Flyweight* fly = fc->GetFlyweight("hello"); 17 18 Flyweight* fly1 = fc->GetFlyweight("hello"); 19 20 //应用外部状态 21 fly->Operation(extrinsicState); 22 23 fc->GetFlyweightCount(); 24 25 return 0; 26 }
另外一个文本处理的例子
参考:
http://www.dofactory.com/Patterns/PatternFlyweight.aspx
http://sourcemaking.com/design_patterns/flyweight/c%2523#
代码如下:
Document.h
1 #ifndef _DOCUMENT_H_ 2 #define _DOCUMENT_H_ 3 4 #include <string> 5 #include <vector> 6 7 using namespace std; 8 9 // The 'Flyweight' abstract class 10 class character 11 { 12 public: 13 //析构函数 14 virtual ~character(); 15 //应用外部状态 16 virtual void Display(int width,int height,int ascent,int descent,int pointSize)=0; 17 //获取内部状态 18 virtual char GetSymbol()=0; 19 protected: 20 /*-----内部状态---------*/ 21 char symbol; 22 /*----------------------/ 23 24 /*-----外部状态---------*/ 25 int width; 26 int height; 27 int ascent; 28 int descent; 29 int pointSize; 30 /*----------------------*/ 31 //构造函数 32 character(char c); 33 }; 34 35 //A 'ConcreteFlyweight' class 36 class characterA:public character 37 { 38 public: 39 characterA(char c); 40 ~characterA(); 41 virtual void Display(int width,int height,int ascent,int descent,int pointSize); 42 virtual char GetSymbol(); 43 protected: 44 private: 45 }; 46 47 //B 'ConcreteFlyweight' class 48 class characterB:public character 49 { 50 public: 51 characterB(char c); 52 ~characterB(); 53 virtual void Display(int width,int height,int ascent,int descent,int pointSize); 54 protected: 55 private: 56 }; 57 58 //C 'ConcreteFlyweight' class 59 class characterC:public character 60 { 61 public: 62 characterC(char c); 63 ~characterC(); 64 virtual void Display(int width,int height,int ascent,int descent,int pointSize); 65 protected: 66 private: 67 }; 68 69 //D 'ConcreteFlyweight' class 70 class characterD:public character 71 { 72 public: 73 characterD(char c); 74 ~characterD(); 75 virtual void Display(int width,int height,int ascent,int descent,int pointSize); 76 protected: 77 private: 78 }; 79 /* 80 ... 81 */ 82 83 //The 'FlyweightFactory' class 84 class characterFactory 85 { 86 public: 87 characterFactory(); 88 ~characterFactory(); 89 //申请一个character对象 90 character* GetCharacter(char); 91 //获取存储的character*数量 92 vector<character*>::size_type GetCount(); 93 private: 94 //保存character*的容器,可以换成List等其它容器 95 vector<character*> m_vecCharacter; 96 }; 97 98 #endif
Document.cpp
1 #include "Document.h" 2 #include <iostream> 3 4 character::character(char c) 5 { 6 this->symbol = c; 7 } 8 9 character::~character() 10 { 11 } 12 13 characterA::characterA(char c):character(c) 14 { 15 } 16 17 characterA::~characterA() 18 { 19 } 20 21 char characterA::GetSymbol() 22 { 23 return this->symbol; 24 } 25 26 void characterA::Display(int width,int height,int ascent,int descent,int pointSize) 27 { 28 //接收并作用外部状态 29 this->ascent = ascent; 30 this->descent = descent; 31 this->height = height; 32 this->pointSize = pointSize; 33 this->width = width; 34 35 cout << this->symbol <<" " 36 << this->ascent <<" " 37 << this->descent <<" " 38 << this->height <<" " 39 << this->pointSize <<" " 40 << this->width << endl; 41 } 42 43 characterFactory::characterFactory() 44 {} 45 46 characterFactory::~characterFactory() 47 {} 48 49 character* characterFactory::GetCharacter(char c) 50 { 51 vector<character*>::iterator iter = this->m_vecCharacter.begin(); 52 for(;iter != this->m_vecCharacter.end();iter++) 53 { 54 if((*iter)->GetSymbol() == c) 55 { 56 return *iter; 57 } 58 } 59 character* pf = new characterA(c); 60 this->m_vecCharacter.push_back(pf); 61 return pf; 62 } 63 64 vector<character*>::size_type characterFactory::GetCount() 65 { 66 return this->m_vecCharacter.size(); 67 }
main.cpp
1 #include "Flyweight.h" 2 #include "Document.h" 3 #include <iostream> 4 #include <string> 5 6 using namespace std; 7 8 int main() 9 { 10 //存储外部状态 11 int ascent = 70; 12 int descent = 0; 13 int height = 100; 14 int width = 120; 15 int pointSize = 10; 16 17 string test = "AABCDDEFGHI"; 18 string::iterator it = test.begin(); 19 characterFactory* pcF = new characterFactory(); 20 for(;it!=test.end();it++) 21 { 22 pointSize++; 23 char c = *it; 24 //申请一个character对象 25 character* charc= pcF->GetCharacter(c); 26 //应用外部状态 27 charc->Display(width,height,ascent,descent,pointSize); 28 } 29 vector<character*>::size_type sizeChar = pcF->GetCount(); 30 cout << "count:" << sizeChar << endl; 31 32 return 0; 33 }