设计模式 | 享元模式

1 | 享元模式的概述

如果一个软件系统在运行时所创建的相同或相似的对象数量太多,将导致运行代价过高,带来系统资源浪费、性能下降等问题。

例如:在一个文本字符串中存在很多重复的字符,如果每一个字符都用一个单独的对象来表示,将会占用较多的内存空间,那么如何避免系统中出现大量相同或相似的对象,同时又不影响客户端程序通过面向对象的方式对这些对象进行操作呢?享元模式正是为解决这―类问题而“诞生”。享元模式通过共享技术实现相同式相创对免的重用,在逻辑上每一个出现的字符都有一个对象与之对应,然而在物理上它们却共享同一个享元对象,这个对象可以出现在一个字符串的不同地方,相同的字符对象都指向同一个实例。在享元模式中,存储这些共享实例对象的地方称为享元池(Flyweight Pool)

用户可以针对每一个不同的字符创建一个享元对象,将其放在享元池中,待需要时再从享元池中取出。字符享元对象示意图如图14-1所示。

享元模式以共享的方式高效地支持大量细粒度对象示意图重用,享元对象能做到共享的关键是区分了内部状态(Intrinsic State)外部状态(Extrinsic State)

  • (1)内部状态是存储在享元对象内部并且不会随环境改变而改变的状态,内部状态可以共享。
  1. 例如字符的内容,不会随外部环境的变化而变化,无论在任何环境下字符“a”始终是“a”,都不会变成“b”。
  • (2)外部状态是随环境改变而改变的、不可以共享的状态。享元对象的外部状态通常由客户端保存,并在享元对象被创建之后,需要使用的时候再传入享元对象内部。一个外部状态与另一个外部状态之间是相互独立的。
  1. 例如字符的颜色,可以在不同的地方有不同的颜色,
  2. 例如有的“a”是红色的,有的“a”是绿色的,字符的大小也是如此,有的“a”是五号字,有的“a”是四号字。而且字符的颜色和大小是两个独立的外部状态,它们可以独立变化,相互之间没有影响,客户端可以在使用时将外部状态注入享元对象中。

正因为区分了内部状态和外部状态,可以将具有相同内部状态的对象存储到享元池中,享元池中的对象是可以实现共享的,需要的时候将对象从享元池中取出,即可实现对象的复用。通过向取出的对象注入不同的外部状态,可以得到一系列相似的对象,而这些对象在内存中实际上只存储一份。

享元模式的定义

  • 享元模式:运用共享技术有效地支持大量细粒度对象的复用。
  • Flyweight Pattern: Use sharing to support large numbers of fine-grained objects efficiently.

享元模式是享元模式要求能够被共享的对象必须是细粒度对象,它又称为轻量级模式,享元模式—种对象结构型模式。

2 | 享元模式的结构与实现

2.1 享元模式的结构

享元模式结构较为复杂,通常结合工厂模式一起使用,在它的结构图中包含了一个享元工厂类,其结构如图14-2所示。

设计模式 | 享元模式_第1张图片

由图14-2可知,享元模式包含以下4个角色。

  • (1) Flyweight(抽象享元类):它通常是一个接口或抽象类,在抽象享元类中声明了具体享元类公共的方法,这些方法可以向外界提供享元对象的内部数据(内部状态),同时也可以通过这些方法设置外部数据(外部状态)。
  • (2) ConcreteFlyweight(具体享元类):它实现了抽象享元类,其实例称为享元对象,并在具体享元类中为内部状态提供了存储空间。通常可以结合单例模式来具体设计享元类,为每一个具体享元类提供唯一的享元对象。
  • (3) UnsharedConcreteFlyweight(非共享具体享元类):并不是所有的抽象享元类的子类都需要被共享,用户可以将不能被共享的子类设计为非共享具体享元类,当需要一个非共享具体享元类的对象时可以直接通过实例化创建。
  • (4) FlyweightFactory(享元工厂类):享元工厂类用于创建并管理享元对象,它针对抽象享元类编程,将各种类型的具体享元对象存储在一个享元池中,享元池一般设计为一个存储“键值对”的集合(也可以是其他类型的集合),可以结合工厂模式进行设计。当用户请求一个具体享元对象时,享元工厂提供一个存储在享元池中已创建的实例或者创建一个新的实例(如果不存在),返回新创建的实例并将其存储在享元池中。

2.2 享元模式的实现

享元模式的典型代码如下:

using System.Collections;

namespace FlyweightPattern
{
    /// 
    /// 抽象享元类
    /// 
    abstract class Flyweight
    {
        public abstract void Operation(string extrinsicState);
    }

    /// 
    /// 具体享元类,要将内部状态和外部状态分开处理
    /// 
    class ConcreteFlyweight : Flyweight
    {
        //【内部状态|intrinsicState】作为成员变量,同一个享元对象其内部状态是一致的
        private string intrinsicState;

        public ConcreteFlyweight(string intrinsicState)
        {
            this.intrinsicState = intrinsicState;
        }

        //【外部状态|extrinsicState】在使用时由外部设置,不保存在享元对象中,即使是同一个对象,在每一次调用时可以传入不同的外部状态
        public override void Operation(string extrinsicState)
        {
            //实现业务方法
        }
    }

    /// 
    /// 非共享具体享元类
    /// 
    class UnsharedConcreteFlyweight : Flyweight
    {
        public override void Operation(string extrinsicState)
        {
            //实现业务方法
        }
    }

    /// 
    /// 享元工厂类
    /// 
    class FlyweightFactory
    {
        //定义一个Hashtable用于存储享元对象,实现享元池
        private readonly Hashtable flyweights = new Hashtable();

        public Flyweight GetFlyweight(string key)
        {
            //如果对象存在,则直接从享元池获取
            if (flyweights.ContainsKey(key))
            {
                return (Flyweight)flyweights[key];
            }
            //如果对象不存在,先创建一个新的对象添加到享元池中,然后返回
            else
            {
                Flyweight fw = new ConcreteFlyweight("state");
                flyweights.Add(key, fw);
                return fw;
            }
        }
    }
}
  • 享元类 Flyweight 的设计是享元模式的要点之一,为了提高系统的可扩展性,通常要定义一个抽象享元类作为所有具体享元类的公共父类;
  • 具体享元类 ConcreteFlyweight  中要将内部状态和外部状态分开处理,通常将内部状态作为具体的享元类的成员变量,而将外部状态通过注入的方式添加到具体享元类中;
  • 非共享具体享元类 UnsharedConcreteFlyweight 是实现无需共享的抽象享元类 Flyweight 的子类;
  • 享元工厂 FlyweightFactory 提供一个用于存储享元对象的享元池,当用户需要对象时,如果享元池中不存在,则创建一个新的享元对象返回给用户,并且在享元池中保存该新增对象;

3 | 享元模式的应用实例

3.1 实例说明

某软件公司要求开发一个围棋软件,其界面效果如下图所示:

设计模式 | 享元模式_第2张图片

该软件公司开发人员通过对围棋软件进行分析发现,在上图中,围棋棋盘中包含大量的黑子和白子,它们的形状、大小都一模一样,只是出现的位置不同而已。如果将一个棋子都作为一个独立的对象存储在内存中,将导致该围棋软件在运行时所需的内存空间较大,如何降低运行代价,提高系统性能是需要解决的一个问题。为了解决该问题现使用享元模式来设计该围棋软件的棋子对象。

3.2 实例类图

设计模式 | 享元模式_第3张图片

3.3 实例代码

从上面的围棋图中分析,其代码结构设计如下:

  • IgoChessman 围棋棋子类,充当抽象享元类;
using System;

namespace FlyweightPattern.IgoChessmanSample
{
    abstract class IgoChessman
    {
        public abstract string GetColor();

        public void Display()
        {
            Console.WriteLine($"棋子颜色:{this.GetColor()}");	
        }
    }
}
  • BlackIgoChessman/ 黑色棋子 和 WhitelgoChessmt/ 白色棋子充当具体享元类;
namespace FlyweightPattern.IgoChessmanSample
{
    class BlackIgoChessman : IgoChessman
    {
        public override string GetColor()
        {
            return "黑色";
        }	
    }
}
  • lgoChessmanFactory 围棋棋子工厂类,充当享元工厂类,适用单例模式对其进行设计;
using System.Collections;

namespace FlyweightPattern.IgoChessmanSample
{
    /// 
    /// 享元工厂 + 单例模式
    /// 
    class IgoChessmanFactory
    {
        #region 单例模式
        private static readonly IgoChessmanFactory instance = new IgoChessmanFactory();
        private readonly Hashtable ht; //使用Hashtable来存储享元对象,充当享元池

        private IgoChessmanFactory()
        {
            ht = new Hashtable();
            IgoChessman black, white;
            black = new BlackIgoChessman();
            ht.Add("b", black);
            white = new WhiteIgoChessman();
            ht.Add("w", white);
        } 
        #endregion

        //返回享元工厂类的唯一实例
        public static IgoChessmanFactory GetInstance()
        {
            return instance;
        }

        //通过key来获取存储在Hashtable中的享元对象
        public IgoChessman GetIgoChessman(string color)
        {
            return (IgoChessman)ht[color];
        }
    }
}
  •  客户端调用

设计模式 | 享元模式_第4张图片

4 | 有外部状态的享元模式

在上面的3.3节应用实例中,对围棋子进一步分析,不难发现,虽然黑色棋子和白色棋子可以共享,但是它们将显示在棋盘的不同位置,如何让相同的黑子或白子能够多次重复显示但位于一个棋盘的不同地方?

解决方法之一

  • 将棋子的位置定义为棋子的一个外部状态,在需要时再进行设置。

设计模式 | 享元模式_第5张图片

改造步骤

  1.  新增一个坐标类 Coordinates ,存储每一个棋子的位置;
  2. 在 IgoChessman 类中新增 Display(Coordinates coord) 重载方法,形参为坐标类型,用于显示棋子时指定其坐标位置;
// 1.新增坐标类
namespace FlyweightPattern.IgoChessmanSample
{
    /// 
    /// 坐标类
    /// 
    class Coordinates
    {
        public Coordinates(int x, int y)
        {
            X = x;
            Y = y;
        }

        public int X { get; set; }
        public int Y { get; set; }
    }
}

// 2.在 IgoChessman 类中添加 Display 重载方法 
public void Display(Coordinates coord)
{
    Console.WriteLine($"棋子颜色:{this.GetColor()},棋子位置:{coord.X},{coord.Y}");
}
  •  客户端调用

设计模式 | 享元模式_第6张图片

完整代码示例请查看=》 https://gitee.com/dolayout/DesignPatternOfCSharp/tree/master/DesignPatternOfCSharp/FlyweightPattern

5 | 单纯享元模式和复合享元模式

标准的享元模式结构图中既包含可以共享的具体享元类,也包含不可以共享的非共享具体享元类。但是在实际使用的过程中,有时候会用到两种特殊的享元模式:

5.1 单纯享元模式

  • 在单纯享元模式中,所有的具体享元类都是可以共享的,不存在非共享具体享元类。单纯享元模式的结构如图如下所示:

设计模式 | 享元模式_第7张图片

 5.2 复合享元模式

  • 将些单纯享元对象使用组合模式加以组合,还可以形成复合享元对象,这样的复合享元对象本身不能共享,但是它们可以分解成单纯享元对象,而后者则可以共享。复合享元模式的结构图如下所示:

设计模式 | 享元模式_第8张图片

通过使用复合享元模式,可以让复合享元类 CompositeConcreteFlyweight 中所包含的每个单纯享元类 ConcreteFlvweioht 都具有相同的外部状态,而这些单纯享元的内部状态往往不同。

如果希望为多个内部状态不同的享元对象设置相同的外部状态,可以考虑使用复合享元模式

 

6 | 享元模式的优缺点与适用环境

当系统中存在大量相同或者相似的对象时,享元模式是一种较好的解决方案,它通过共享技术实现相同或相似的细粒度对象的复用,从而节约了内存空间,提高了系统性能。相比其他结构型设计模式,享元模式的使用频率并不算太高,但是作为一种以“节约内存,提高性能”为出发点的设计模式,它在软件开发中还是得到了一定程度的应用。

6.1 享元模式的优点

  • (1)享元模式可以减少内存中对象的数量,使得相同或者相似的对象在内存中只保存一份,从而可以节约系统资源,提高系统性能。
  • (2)在享元模式中,外部状态相对独立,而且不会影响其内部状态,从而使得享元对复可以在不同的环境中被共享。

6.2 享元模式的缺点

  • (1)享元模式使得系统变得复杂,需要分离出内部状态和外部状态,这使得程序的逻辑复杂化。
  • (2)为了使对象可以共享,享元模式需要将享元对象的部分状态外部化,而读取外部状态将使得运行时间变长。

6.3 享元模式的适用环境

  • (1) 一个系统有大量相同或者相似的对象,造成了内存的大量耗费。
  • (2) 对象的大部分状态都可以外部化,可以将这些外部状态传入对象中。
  • (3) 在使用享元模式时需要维护一个存储享元对象的享元池,而这需要耗费一定的系统资源,因此,在需要多次重复使用享元对象时才值得使用享元模式。

你可能感兴趣的:(设计模式(Design,pattern),Flyweight,享元模式,Pattern)