一、引言
在软件开发过程,如果我们需要重复使用某个对象的时候,若重复地使用new创建这个对象的话,就需要多次地去申请内存空间了,这样可能会出现内存使用越来越多的情况,这样的问题是非常严重。享元模式可以解决这个问题,下面具体看看享元模式是如何去解决这个问题的。
二、享元模式的详细介绍
在前面说了,享元模式可以解决上面的问题了。在介绍享元模式之前,让我们先要分析下如果去解决上面那个问题。上面的问题就是重复创建了同一个对象,如果让我们去解决这个问题肯定会这样想:“既然都是同一个对象,能不能只创建一个对象,然后下次需要创建这个对象的时候,让它直接用已经创建好了的对象就好了“。也就是说——让一个对象共享。不错,这个也是享元模式的实现精髓所在。
2.1 定义
介绍完享元模式的精髓之后,让我们具体看看享元模式的正式定义:
享元模式——运用共享技术有效地支持大量细粒度的对象。享元模式可以避免大量相似类的开销。在软件开发中如果需要生成大量细粒度的类实例来表示数据,这些实例除了几个参数外基本上都是相同的,这时候就可以使用享元模式来大幅度减少需要实例化类的数量。如果能把这些参数(指的这些类实例不同的参数)移动类实例外面,在方法调用时将他们传递进来,这样就可以通过共享,达到大幅度地减少单个实例的数目。(这个也是享元模式的实现要领),然而我们把类实例外面的参数称为享元对象的外部状态,把在享元对象内部定义称为内部状态。
具体享元对象的内部状态与外部状态的定义为:
内部状态:在享元对象的内部并且不会随着环境的改变而改变的共享部分。
外部状态:随环境改变而改变的,不可以共享的状态。
2.2 享元模式实现
分析完享元模式的实现思路之后,相信大家实现享元模式肯定没什么问题了,下面以一个实际的应用来实现下享元模式。这个例子是:一个文本编辑器中会出现很多字面,使用享元模式去实现这个文本编辑器的话,会把每个字面做成一个享元对象。享元对象的内部状态就是这个字面,而字母在文本中的位置和字体风格等其他信息就是它的外部状态。下面就以这个例子来实现下享元模式,具体实现代码如下:
////// 客户端调用 /// class Client { static void Main(string[] args) { // 定义外部状态,例如字母的位置等信息 int externalstate = 10; // 初始化享元工厂 FlyweightFactory factory = new FlyweightFactory(); // 判断是否已经创建了字母A,如果已经创建就直接使用创建的对象A Flyweight fa = factory.GetFlyweight("A"); if (fa != null) { // 把外部状态作为享元对象的方法调用参数 fa.Operation(--externalstate); } // 判断是否已经创建了字母B Flyweight fb = factory.GetFlyweight("B"); if (fb != null) { fb.Operation(--externalstate); } // 判断是否已经创建了字母C Flyweight fc = factory.GetFlyweight("C"); if (fc != null) { fc.Operation(--externalstate); } // 判断是否已经创建了字母D Flyweight fd= factory.GetFlyweight("D"); if (fd != null) { fd.Operation(--externalstate); } else { Console.WriteLine("驻留池中不存在字符串D"); // 这时候就需要创建一个对象并放入驻留池中 ConcreteFlyweight d = new ConcreteFlyweight("D"); factory.flyweights.Add("D", d); } Console.Read(); } } ////// 享元工厂,负责创建和管理享元对象 /// public class FlyweightFactory { // 最好使用泛型Dictionary//public Dictionary flyweights = new Dictionary (); public Hashtable flyweights = new Hashtable(); public FlyweightFactory() { flyweights.Add("A", new ConcreteFlyweight("A")); flyweights.Add("B", new ConcreteFlyweight("B")); flyweights.Add("C", new ConcreteFlyweight("C")); } public Flyweight GetFlyweight(string key) { // 更好的实现如下 //Flyweight flyweight = flyweights[key] as Flyweight; //if (flyweight == null) //{ // Console.WriteLine("驻留池中不存在字符串" + key); // flyweight = new ConcreteFlyweight(key); //} //return flyweight; return flyweights[key] as Flyweight; } } /// /// 抽象享元类,提供具体享元类具有的方法 /// public abstract class Flyweight { public abstract void Operation(int extrinsicstate); } // 具体的享元对象,这样我们不把每个字母设计成一个单独的类了,而是作为把共享的字母作为享元对象的内部状态 public class ConcreteFlyweight : Flyweight { // 内部状态 private string intrinsicstate ; // 构造函数 public ConcreteFlyweight(string innerState) { this.intrinsicstate = innerState; } ////// 享元类的实例方法 /// /// 外部状态 public override void Operation(int extrinsicstate) { Console.WriteLine("具体实现类: intrinsicstate {0}, extrinsicstate {1}", intrinsicstate, extrinsicstate); } }
在享元模式的实现中,我们没有像之前一样,把一个细粒度的类实例设计成一个单独的类,而是把它作为共享对象的内部状态放在共享类的内部定义,具体的解释注释中都有了,大家可以参考注释去进一步理解享元模式。
2.3 享元模式的类图
看完享元模式的实现之后,为了帮助大家理清楚享元模式中各类之间的关系,下面给出上面实现代码中的类图,如下所示:
在上图中,涉及的角色如下几种角色:
1)抽象享元角色(Flyweight):此角色是所有的具体享元类的基类,为这些类规定出需要实现的公共接口。那些需要外部状态的操作可以通过调用方法以参数形式传入。
2)具体享元角色(ConcreteFlyweight):实现抽象享元角色所规定的接口。如果有内部状态的话,可以在类内部定义。
3)享元工厂角色(FlyweightFactory):本角色复杂创建和管理享元角色。本角色必须保证享元对象可以被系统适当地共享,当一个客户端对象调用一个享元对象的时候,享元工厂角色检查系统中是否已经有一个符合要求的享元对象,如果已经存在,享元工厂角色就提供已存在的享元对象,如果系统中没有一个符合的享元对象的话,享元工厂角色就应当创建一个合适的享元对象。
4)客户端角色(Client):本角色需要存储所有享元对象的外部状态。
三、享元模式的优缺点
分析完享元模式的实现之后,让我们继续分析下享元模式的优缺点:
优点:
1)降低了系统中对象的数量,从而降低了系统中细粒度对象给内存带来的压力。
缺点:
1)为了使对象可以共享,需要将一些状态外部化,这使得程序的逻辑更复杂,使系统复杂化。
2)享元模式将享元对象的状态外部化,而读取外部状态使得运行时间稍微变长。
四、使用场景
在下面所有条件都满足时,可以考虑使用享元模式:
1)一个系统中有大量的对象;
2)这些对象耗费大量的内存;
3)这些对象中的状态大部分都可以被外部化;
4)这些对象可以按照内部状态分成很多的组,当把外部对象从对象中剔除时,每一个组都可以仅用一个对象代替;
5)软件系统不依赖这些对象的身份;
6)满足上面的条件的系统可以使用享元模式。但是使用享元模式需要额外维护一个记录子系统已有的所有享元的表,而这也需要耗费资源,所以,应当在有足够多的享元实例可共享时才值得使用享元模式。
注:在.NET类库中,string类的实现就使用了享元模式,更多内容可以参考字符串驻留池的介绍,同时也可以参考这个博文深入理解.NET中string类的设计——http://www.cnblogs.com/artech/archive/2010/11/25/internedstring.html
五、总结
到这里,享元模式的介绍就结束了,享元模式主要用来解决由于大量的细粒度对象所造成的内存开销的问题,它在实际的开发中并不常用,可以作为底层的提升性能的一种手段。
备注:
既然已经到享元模式了,那对于了解单例模式的同学,就会产生一个问题,这两种模式有什么区别:
1、单例模式是类级别的。
一个类只能有一个对象实例,没有办法创建多个对象实例的。
此种模式主要是为了共享数据或状态的目的。
2、享元模式是对象级别的。
话说一个类可以创建多个对象,但是为了提高系统性能,降低内存消耗,所以才采用的享元模式。
可以将此模式理解成一组共享的对象集合。
享元模式可以再次创建对象,也可以取缓存对象。所以可以在需要的时候创建更多的对象。
此种模式虽然节省了很多的内存空间,避免了资源浪费。但是,对于高并发的情况不是线程安全的。试想一下,同一时间获取到了对象,传入了两个不同内容的外部实例,这时候会产生内容覆盖的情况。所以,对于涉及到如数据库操作的类库,应该避免使用享元模式。
3、两种模式的目的是一样的,都是为了减少内存的开销。享元模式也可以看成是单例模式的一种扩展。
参考链接: