C#设计模式之十一享元模式(Flyweight Pattern)【结构型】

简介: 原文:C#设计模式之十一享元模式(Flyweight Pattern)【结构型】一、引言   今天我们要讲【结构型】设计模式的第六个模式,该模式是【享元模式】,英文名称是:Flyweight Pattern。
原文: C#设计模式之十一享元模式(Flyweight Pattern)【结构型】

一、引言

   今天我们要讲【结构型】设计模式的第六个模式,该模式是【享元模式】,英文名称是:Flyweight Pattern。还是老套路,先从名字上来看看。“享元”是不是可以这样理解,共享“单元”,单元是什么呢,举例说明,对于图形而言就是图元,对于英文来说就只26个英文字母,对于汉语来说就是每个汉字,也可以这样理解“元”,构成事物的最小单元,这些单元如果大量、且重复出现,可以缓存重复出现的单元,达到节省内存的目的,换句说法就是享元是为了节省空间,对于计算机而言就是内存。面向对象很好地解决了系统抽象性的问题(系统抽象性指把系统里面的事物写成类,类可以实例化成为对象,用对象和对象之间的关系来设计系统),在大多数情况下,这样做是不会损及系统的性能的。但是,在某些特殊的应用中,由于对象的数量太大,并且这些大量的对象中有很多是重复的,如果每个对象都单独的创建(C#的语法是new)出来,会给系统带来难以承受的内存开销。比如图形应用中的图元等对象、字处理应用中的字符对象等。

二、享元模式的详细介绍

2.1、动机(Motivate)

   在软件系统中,采用纯粹对象方案的问题在于大量细粒度的对象会很快充斥在系统中,从而带来很高的运行时代价——主要指内存需求方面的代价。如何在避免大量细粒度对象问题的同时,让外部客户程序仍然能够透明地使用面向对象的方式来进行操作?

2.2、意图(Intent)

   运用共享技术有效地支持大量细粒度的对象。                                             ——《设计模式》GoF

2.3、结构图(Structure)

       i

2.4、模式的组成
    
    (1)、抽象享元角色(Flyweight):此角色是所有的具体享元类的基类,为这些类规定出需要实现的公共接口。那些需要外部状态的操作可以通过调用方法以参数形式传入。

    (2)、具体享元角色(ConcreteFlyweight):实现抽象享元角色所规定的接口。如果有内部状态的话,可以在类内部定义。

    (3)、享元工厂角色(FlyweightFactory):本角色负责创建和管理享元角色。本角色必须保证享元对象可以被系统适当地共享,当一个客户端对象调用一个享元对象的时候,享元工厂角色检查系统中是否已经有一个符合要求的享元对象,如果已经存在,享元工厂角色就提供已存在的享元对象,如果系统中没有一个符合的享元对象的话,享元工厂角色就应当创建一个合适的享元对象。

    (4)、客户端角色(Client):本角色需要存储所有享元对象的外部状态。

2.5、享元模式的具体代码实现

    说起“享元模式”,我这里有一个很好的场景可以进行说明。我们知道在战斗的游戏场景中,会有很多战士,基本上战士都是差不多的,小区别战士忽略,最大的区别就是拿的武器不同而已。在大型的战争游戏中,会有大量的士兵出来战斗,我们写程序的时候就可以用“享元”来解决大量战士的情况。

  1 namespace 享元模式的实现
  2     {
  3         /// <summary>
  4         /// 享元模式不是很难,但是有些状态需要单独处理,以下就是该模式的C#实现,有些辅助类,大家应该看得出吧,别混了。
  5         /// </summary>
  6         class Client
  7         {
  8             static void Main(string[] args)
  9             {
 10                 //比如,我们现在需要10000个一般士兵,只需这样
 11                 SoldierFactory factory = new SoldierFactory();
 12                 AK47 ak47 = new AK47();
 13                 for (int i = 0; i < 100; i++)
 14                 {
 15                     Soldier soldier = null;
 16                     if (i <= 20)
 17                     {
 18                         soldier = factory.GetSoldier("士兵" + (i + 1), ak47, SoldierType.Normal);
 19                     }
 20                     else
 21                     {
 22                         soldier = factory.GetSoldier("士兵" + (i + 1), ak47, SoldierType.Water);
 23                     }     
 24                     soldier.Fight();
 25                 }
 26                 //我们有这么多的士兵,但是使用的内存不是很多,因为我们缓存了。
 27                 Console.Read();
 28             }
 29         }
 30 
 31         //这些是辅助类型
 32         public enum SoldierType
 33         {
 34             Normal,
 35             Water
 36         }
 37 
 38         //该类型就是抽象战士Soldier--该类型相当于抽象享元角色
 39         public abstract class Soldier
 40         {
 41             //通过构造函数初始化士兵的名称
 42             protected Soldier(string name)
 43             {
 44                 this.Name = name;
 45             }
 46 
 47             //士兵的名字
 48             public string Name { get; private set; }
 49 
 50             //可以传入不同的武器就用不同的活力---该方法相当于抽象Flyweight的Operation方法
 51             public abstract void Fight();
 52 
 53             public Weapen WeapenInstance { get; set; }
 54         }
 55 
 56         //一般类型的战士,武器就是步枪---相当于具体的Flyweight角色
 57         public sealed class NormalSoldier : Soldier
 58         {
 59             //通过构造函数初始化士兵的名称
 60             public NormalSoldier(string name) : base(name) { }
 61 
 62             //执行享元的方法---就是Flyweight类型的Operation方法
 63             public override void Fight()
 64             {
 65                 WeapenInstance.Fire("士兵:"+Name+" 在陆地执行击毙任务");
 66             }
 67         }
 68 
 69         //这是海军陆战队队员,武器精良----相当于具体的Flyweight角色
 70         public sealed class WaterSoldier : Soldier
 71         {
 72             //通过构造函数初始化士兵的名称
 73             public WaterSoldier(string name) : base(name) { }
 74 
 75             //执行享元的方法---就是Flyweight类型的Operation方法
 76             public override void Fight()
 77             {
 78                 WeapenInstance.Fire("士兵:"+Name+" 在海中执行击毙任务");
 79             }
 80         }
 81 
 82         //此类型和享元没太大关系,可以算是享元对象的状态吧,需要从外部定义
 83         public abstract class Weapen
 84         {
 85             public abstract void Fire(string jobName);
 86         }
 87 
 88         //此类型和享元没太大关系,可以算是享元对象的状态吧,需要从外部定义
 89         public sealed class AK47:Weapen
 90         {
 91             public override void Fire(string jobName)
 92             {
 93                 Console.WriteLine(jobName);
 94             }
 95         }
 96 
 97         //该类型相当于是享元的工厂---相当于FlyweightFactory类型
 98         public sealed class SoldierFactory
 99         {
100             private static IList<Soldier> soldiers;
101 
102             static SoldierFactory()
103             {
104                 soldiers = new List<Soldier>();
105             }
106 
107             Soldier mySoldier = null;
108             //因为我这里有两种士兵,所以在这里可以增加另外一个参数,士兵类型,原模式里面没有,
109             public Soldier GetSoldier(string name, Weapen weapen, SoldierType soldierType)
110             {
111                 foreach (Soldier soldier in soldiers)
112                 {
113                     if (string.Compare(soldier.Name, name, true) == 0)
114                     {
115                         mySoldier = soldier;
116                         return mySoldier;
117                     }
118                 }
119                 //我们这里就任务名称是唯一的
120                 if (soldierType == SoldierType.Normal)
121                 {
122                     mySoldier = new NormalSoldier(name);
123                 }
124                 else
125                 {
126                     mySoldier = new WaterSoldier(name);
127                 }
128                 mySoldier.WeapenInstance = weapen;
129 
130                 soldiers.Add(mySoldier);
131                 return mySoldier;
132             }
133         }
134     }


   这个模式很简单,就话不多说了。

三、享元模式的实现要点:
    
    面向对象很好地解决了抽象性的问题,但是作为一个运行在机器中的程序实体,我们需要考虑对象的代价问题。Flyweight设计模式主要解决面向对象的代价问题,一般不触及面向对象的抽象性问题。

    Flyweight采用对象共享的做法来降低系统中对象的个数,从而降低细粒度对象给系统带来的内存压力。在具体实现方面,要注意对象状态的处理。

 对象的数量太大从而导致对象内存开销加大——什么样的数量才算大?这需要我们仔细的根据具体应用情况进行评估,而不能凭空臆断。

    3.1】、享元模式的优点

      (1)、享元模式的优点在于它能够极大的减少系统中对象的个数。

      (2)、享元模式由于使用了外部状态,外部状态相对独立,不会影响到内部状态,所以享元模式使得享元对象能够在不同的环境被共享。

    3.2】、享元模式的缺点

      (1)、由于享元模式需要区分外部状态和内部状态,使得应用程序在某种程度上来说更加复杂化了。

      (2)、为了使对象可以共享,享元模式需要将享元对象的状态外部化,而读取外部状态使得运行时间变

    3.3】、在下面所有条件都满足时,可以考虑使用享元模式:

       (1)、一个系统中有大量的对象;

       (2)、这些对象耗费大量的内存;

       (3)、这些对象中的状态大部分都可以被外部化

       (4)、这些对象可以按照内部状态分成很多的组,当把外部对象从对象中剔除时,每一个组都可以仅用一个对象代替软件系统不依赖这些对象的身份,

        满足上面的条件的系统可以使用享元模式。但是使用享元模式需要额外维护一个记录子系统已有的所有享元的表,而这也需要耗费资源,所以,应当在有足够多的享元实例可共享时才值得使用享元模式。

四、.NET 中享元模式的实现

    .NET在C#中有一个Code Behind机制,它表面有一个aspx文件,背后又有一个cs文件,它的编译过程实际上会把aspx文件解析成C#文件,然后编译成dll,在这个过程中,我们在aspx中写的任何html代码都会转化为literal control,literal control是一个一般的文本控件,它就表示html标记。当这些标记有一样的时候,构建控件树的时候就会用到Flyweight模式.

  它的应用并不是那么平凡,只有在效率空间确实不高的时候我们才用它。

五、总结

    刚开始接触这个模式的时候,感觉这个模式不是特别难,在我们编码的过程中也有涉及,但是在学习的过程中也走了不少弯路,任何设计模式都有他特定的使用场景,小心误用。这个模式在业务系统中相对而言使用的并不多,在类似游戏场景中、字符处理等系统用的比较多。还是老话,通过迭代来使用模式,别为了模式而模式。今天就到这里,以后继续。

目录
相关文章
|
2月前
|
设计模式 安全 Java
C# 一分钟浅谈:设计模式之单例模式
【10月更文挑战第9天】单例模式是软件开发中最常用的设计模式之一,旨在确保一个类只有一个实例,并提供一个全局访问点。本文介绍了单例模式的基本概念、实现方式(包括饿汉式、懒汉式和使用 `Lazy&lt;T&gt;` 的方法)、常见问题(如多线程和序列化问题)及其解决方案,并通过代码示例详细说明了这些内容。希望本文能帮助你在实际开发中更好地应用单例模式,提高代码质量和可维护性。
47 1
|
3月前
|
设计模式
设计模式-工厂模式 Factory Pattern(简单工厂、工厂方法、抽象工厂)
这篇文章详细解释了工厂模式,包括简单工厂、工厂方法和抽象工厂三种类型。每种模式都通过代码示例展示了其应用场景和实现方法,并比较了它们之间的差异。简单工厂模式通过一个工厂类来创建各种产品;工厂方法模式通过定义一个创建对象的接口,由子类决定实例化哪个类;抽象工厂模式提供一个创建相关或依赖对象家族的接口,而不需要明确指定具体类。
设计模式-工厂模式 Factory Pattern(简单工厂、工厂方法、抽象工厂)
|
3月前
|
设计模式 C# 开发者
C#设计模式入门实战教程
C#设计模式入门实战教程
|
3月前
|
设计模式 Java
设计模式--适配器模式 Adapter Pattern
这篇文章介绍了适配器模式,包括其基本介绍、工作原理以及类适配器模式、对象适配器模式和接口适配器模式三种实现方式。
|
4月前
|
设计模式 算法 C#
C#设计模式之策略模式
C#设计模式之策略模式
87 19
|
4月前
|
设计模式 安全 程序员
C#设计模式之单例模式
C#设计模式之单例模式
52 3
|
5月前
|
设计模式 存储 C#
|
5月前
|
设计模式 算法 C#
23种设计模式【C#代码举例】(上)
23种设计模式【C#代码举例】(上)
|
6月前
|
设计模式
设计模式-05建造者模式(Builder Pattern)
设计模式-05建造者模式(Builder Pattern)
|
6月前
|
设计模式 C#
技术经验分享:C#设计模式
技术经验分享:C#设计模式
30 0