Net设计模式实例之享元模式( Flyweight Pattern)

简介:

一、享元模式简介(Brief Introduction

享元模式(Flyweight Pattern,运用共享技术有效支持大量细粒度的对象。

Use sharing to support large numbers of fine-grained objects efficiently.

享元模式可以避免大量非常相似类的开销。在程序设计中有时需要生成大量细粒度的类实例来表示数据。如果发现这些实例除了几个参数外基本伤都是相同的,有时就能够受大幅度第减少需要实例化的类的数量。如果能把这些参数移到类实例外面,在方法调用时将他们传递进来,就可以通过共享大幅度地减少单个实例的数目。

享元对象的内部状态与外部状态:

内部状态,在享元对象的内部并且不会随环境改变而改变的共享部分。

外部状态,随环境改变而改变的,不可以共享的状态。

二、解决的问题(What To Solve

       如果一个应用程序使用了大量的对象,而大量的这些对象造成了很大的存储开销,这时可以考虑使用享元模式。

       当对象的大多数状态是外部状态,如果删除对象的外部状态,那么可以用相对较少的共享对象取代很多组对象,这时也可以考虑使用享元模式。

三、享元模式分析(Analysis

1、享元模式结构

 

FlyweightFactory:享元工厂,用来创建和管理Flyweight对象。如果请求的Flyweight对象存在,怎返回已经存在的对象。否则新创建一个新的对象返回。

Flyweight:享元抽象类,通过这个接口,Flyweight可以接受并作用与外部状态。

UnsharedConcreteFlyweight:不需要共享的Flyweight子类。Flyweight接口并不强制共享。

ConcreteFlyweight:实现享元抽象类,为内部状态添加存储空间。

2、代码

1、享元工厂类FlyweightFactory

public class FlyweightFactory

{

    public Hashtable flyweights = new Hashtable();

 

    public FlyweightFactory()

    {

        flyweights.Add("A"new ConcreteFlyweight());

        flyweights.Add("B"new ConcreteFlyweight());

        flyweights.Add("C"new ConcreteFlyweight());

    }

 

    public Flyweight GetFlyweight(string key)

    {

        return flyweights[key] as Flyweight;

    }

}

 

2、享元抽象类Flyweight 及其具体实现类UnsharedConcreteFlyweight

ConcreteFlyweight

public abstract class Flyweight

{

    public abstract void Operation(int extrinsicstate);

}

 

public class UnsharedConcreteFlyweight:Flyweight

{

    public override void Operation(int extrinsicstate)

    {

        Console.WriteLine("{0}:{1}",this.GetType().Name,extrinsicstate);

    }

}

 

public class ConcreteFlyweight:Flyweight

{

    public override void Operation(int extrinsicstate)

    {

        Console.WriteLine("{0}:{1}"this.GetType().Name, extrinsicstate);

    }

}

 

3、客户端代码

static void Main(string[] args)

{

    // Arbitrary extrinsic state

    int extrinsicstate = 20;

    FlyweightFactory factory = new FlyweightFactory();

 

    // Work with different flyweight instances

    Flyweight fx = factory.GetFlyweight("A");

    fx.Operation(--extrinsicstate);

 

    Flyweight fy = factory.GetFlyweight("B");

    fy.Operation(--extrinsicstate);

 

    Flyweight fz = factory.GetFlyweight("C");

    fz.Operation(--extrinsicstate);

 

    UnsharedConcreteFlyweight fu = new UnsharedConcreteFlyweight();

    fu.Operation(--extrinsicstate);

 

    Console.ReadKey();

}

3、实例运行结果

 

四.享元模式实例分析(Example

1、场景

一个文档Document中只有少数字符需要共享。结构如下图所示

 

CharacterFactory,享元工厂,用来创建和管理Charactor对象。如果请求的Charactor对象存在,怎返回已经存在的对象。否则新创建一个新的对象返回。

Character:享元抽象类,通过这个接口,Character可以接受并作用与外部状态。
CharacterA /CharacterB/CharacterC :实现享元抽象类,为内部状态添加存储空间。

2、代码

1字符工厂类CharacterFactory

class CharacterFactory

{

    private Dictionary<charCharacter> _characters = new Dictionary<char,Character>();

    public Character GetCharacter(char key)

    {

        // Uses "lazy initialization"

        Character character = null;

        if (_characters.ContainsKey(key))

        {

            character = _characters[key];

        }

        else

        {

            switch (key)

            {

                case 'A': character = new CharacterA(); break;

                case 'B': character = new CharacterB(); break;

                //...

                case 'Z': character = new CharacterZ(); break;

            }

            _characters.Add(key, character);

        }

        return character;

    }

}

 

2抽象数据对象类DataObject及其具体实现类CustomersData

/// <summary>

/// The 'Flyweight' abstract class

/// </summary>

abstract class Character

{

    protected char symbol;

    protected int width;

    protected int height;

    protected int ascent;

    protected int descent;

    protected int pointSize;

    public abstract void Display(int pointSize);

}

/// <summary>

/// A 'ConcreteFlyweight' class

/// </summary>

class CharacterA : Character

{

    public CharacterA()

    {

        this.symbol = 'A';

        this.height = 100;

        this.width = 120;

        this.ascent = 70;

        this.descent = 0;

    }

    public override void Display(int pointSize)

    {

        this.pointSize = pointSize;

        Console.WriteLine(this.symbol + " (pointsize " + this.pointSize +")");

    }

}

/// <summary>

/// A 'ConcreteFlyweight' class

/// </summary>

class CharacterB : Character

{

    public CharacterB()

    {

        this.symbol = 'B';

        this.height = 100;

        this.width = 140;

        this.ascent = 72;

        this.descent = 0;

    }

    public override void Display(int pointSize)

    {

        this.pointSize = pointSize;

        Console.WriteLine(this.symbol + " (pointsize " + this.pointSize +")");

    }

}

// ... C, D, E, etc.

/// <summary>

/// A 'ConcreteFlyweight' class

/// </summary>

class CharacterZ : Character

{

    // Constructor

    public CharacterZ()

    {

        this.symbol = 'Z';

        this.height = 100;

        this.width = 100;

        this.ascent = 68;

        this.descent = 0;

    }

    public override void Display(int pointSize)

    {

        this.pointSize = pointSize;

        Console.WriteLine(this.symbol +" (pointsize " + this.pointSize +")");

    }

}

 

3、客户端代码

static void Main(string[] args)

{

    // Build a document with text

    string document = "AAZZBBZB";

    char[] chars = document.ToCharArray();

    CharacterFactory factory = new CharacterFactory();

    // extrinsic state

    int pointSize = 10;

    // For each character use a flyweight object

    foreach (char c in chars)

    {

        pointSize++;

        Character character = factory.GetCharacter(c);

        character.Display(pointSize);

    }

    Console.ReadKey();

}

3、实例运行结果

五、总结(Summary

本文对享元模式(Flyweight Pattern)的概念、设计结构图、代码、使用场景、进行了描述。以一个享元模式实例进行了说明。如果一个应用程序使用了大量的对象,而大量的这些对象造成了很大的存储开销,这时可以考虑使用享元模式。



 本文转自灵动生活博客园博客,原文链接: http://www.cnblogs.com/ywqu/archive/2010/01/21/1653087.html ,如需转载请自行联系原作者

相关文章
|
27天前
|
设计模式 存储 Java
【十】设计模式~~~结构型模式~~~享元模式(Java)
文章详细介绍了享元模式(Flyweight Pattern),这是一种对象结构型模式,通过共享技术实现大量细粒度对象的重用,区分内部状态和外部状态来减少内存中对象的数量,提高系统性能。通过围棋棋子的设计案例,展示了享元模式的动机、定义、结构、优点、缺点以及适用场景,并探讨了单纯享元模式和复合享元模式以及与其他模式的联用。
【十】设计模式~~~结构型模式~~~享元模式(Java)
|
2月前
|
设计模式 存储 JavaScript
js设计模式【详解】—— 享元模式
js设计模式【详解】—— 享元模式
40 6
|
3月前
|
设计模式 安全 Java
Java中的单例模式是一种设计模式,它保证一个类只有一个实例,并提供一个全局访问点
Java单例模式确保类仅有一个实例,并提供全局访问点。常见实现包括: - 饿汉式:静态初始化,线程安全。 - 懒汉式:延迟初始化,需同步保证线程安全。 - 双重检查锁定:优化懒汉式,减少同步开销。 - 静态内部类:延迟加载,线程安全。 - 枚举:简洁线程安全,不适用于复杂构造。 - 容器实现:如Spring框架,用于依赖注入。选择依据需求,如延迟加载、线程安全和扩展性。
61 10
|
3月前
|
设计模式 缓存 Java
Java设计模式:享元模式实现高效对象共享与内存优化(十一)
Java设计模式:享元模式实现高效对象共享与内存优化(十一)
|
3月前
|
设计模式 存储 Java
Java设计模式之享元模式详解
Java设计模式之享元模式详解
|
2月前
|
设计模式 缓存 JavaScript
js设计模式实例
【7月更文挑战第2天】JavaScript设计模式包含工厂、单例、建造者、抽象工厂和代理模式等,它们是最佳实践和可重用模板,解决创建、职责分配和通信等问题。例如,工厂模式封装对象创建,单例确保全局唯一实例,建造者模式用于复杂对象构建,抽象工厂创建相关对象集合,而代理模式则控制对象访问。这些模式提升代码质量、可读性和灵活性,是高效开发的关键。
27 0
|
3月前
|
设计模式
设计模式-05建造者模式(Builder Pattern)
设计模式-05建造者模式(Builder Pattern)
|
3月前
|
设计模式
享元模式-大话设计模式
享元模式-大话设计模式
|
4月前
|
设计模式 安全 Java
【设计模式】JAVA Design Patterns——Curiously Recurring Template Pattern(奇异递归模板模式)
该文介绍了一种C++的编程技巧——奇异递归模板模式(CRTP),旨在让派生组件能继承基本组件的特定功能。通过示例展示了如何创建一个`Fighter`接口和`MmaFighter`类,其中`MmaFighter`及其子类如`MmaBantamweightFighter`和`MmaHeavyweightFighter`强制类型安全,确保相同重量级的拳手之间才能进行比赛。这种设计避免了不同重量级拳手间的错误匹配,编译时会报错。CRTP适用于处理类型冲突、参数化类方法和限制方法只对相同类型实例生效的情况。
【设计模式】JAVA Design Patterns——Curiously Recurring Template Pattern(奇异递归模板模式)
|
4月前
|
设计模式 Java 开发者
【搞懂设计模式】享元模式:共享节约,皆大欢喜!
【搞懂设计模式】享元模式:共享节约,皆大欢喜!
62 0