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

简介:

一、享元模式简介(Brief Introduction

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

Use sharing to support large numbers of fine-g rain ed 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 = nu ll ;

        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.ToCharAr ray ();

    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 ,如需转载请自行联系原作者

相关文章
|
8月前
|
设计模式 算法 Java
行为型设计模式-策略模式(Strategy Pattern)
行为型设计模式-策略模式(Strategy Pattern)
|
4月前
|
设计模式
设计模式-工厂模式 Factory Pattern(简单工厂、工厂方法、抽象工厂)
这篇文章详细解释了工厂模式,包括简单工厂、工厂方法和抽象工厂三种类型。每种模式都通过代码示例展示了其应用场景和实现方法,并比较了它们之间的差异。简单工厂模式通过一个工厂类来创建各种产品;工厂方法模式通过定义一个创建对象的接口,由子类决定实例化哪个类;抽象工厂模式提供一个创建相关或依赖对象家族的接口,而不需要明确指定具体类。
设计模式-工厂模式 Factory Pattern(简单工厂、工厂方法、抽象工厂)
|
4月前
|
设计模式 Java
设计模式--适配器模式 Adapter Pattern
这篇文章介绍了适配器模式,包括其基本介绍、工作原理以及类适配器模式、对象适配器模式和接口适配器模式三种实现方式。
|
8月前
|
设计模式 算法
设计模式 - 行为型模式_ 访问者模式Visitor Pattern
设计模式 - 行为型模式_ 访问者模式Visitor Pattern
78 1
设计模式 - 行为型模式_ 访问者模式Visitor Pattern
|
7月前
|
设计模式
设计模式-05建造者模式(Builder Pattern)
设计模式-05建造者模式(Builder Pattern)
|
8月前
|
设计模式 安全 Java
【设计模式】JAVA Design Patterns——Curiously Recurring Template Pattern(奇异递归模板模式)
该文介绍了一种C++的编程技巧——奇异递归模板模式(CRTP),旨在让派生组件能继承基本组件的特定功能。通过示例展示了如何创建一个`Fighter`接口和`MmaFighter`类,其中`MmaFighter`及其子类如`MmaBantamweightFighter`和`MmaHeavyweightFighter`强制类型安全,确保相同重量级的拳手之间才能进行比赛。这种设计避免了不同重量级拳手间的错误匹配,编译时会报错。CRTP适用于处理类型冲突、参数化类方法和限制方法只对相同类型实例生效的情况。
|
8月前
|
设计模式 存储 Java
Java设计模式:解释一下单例模式(Singleton Pattern)。
`Singleton Pattern`是Java中的创建型设计模式,确保类只有一个实例并提供全局访问点。它通过私有化构造函数,用静态方法返回唯一的实例。类内静态变量存储此实例,对外仅通过静态方法访问。
55 1
|
3天前
|
设计模式 前端开发 搜索推荐
前端必须掌握的设计模式——模板模式
模板模式(Template Pattern)是一种行为型设计模式,父类定义固定流程和步骤顺序,子类通过继承并重写特定方法实现具体步骤。适用于具有固定结构或流程的场景,如组装汽车、包装礼物等。举例来说,公司年会节目征集时,蜘蛛侠定义了歌曲的四个步骤:前奏、主歌、副歌、结尾。金刚狼和绿巨人根据此模板设计各自的表演内容。通过抽象类定义通用逻辑,子类实现个性化行为,从而减少重复代码。模板模式还支持钩子方法,允许跳过某些步骤,增加灵活性。
|
2月前
|
设计模式 安全 Java
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
Kotlin教程笔记(51) - 改良设计模式 - 构建者模式
|
4月前
|
设计模式 数据库连接 PHP
PHP中的设计模式:提升代码的可维护性与扩展性在软件开发过程中,设计模式是开发者们经常用到的工具之一。它们提供了经过验证的解决方案,可以帮助我们解决常见的软件设计问题。本文将介绍PHP中常用的设计模式,以及如何利用这些模式来提高代码的可维护性和扩展性。我们将从基础的设计模式入手,逐步深入到更复杂的应用场景。通过实际案例分析,读者可以更好地理解如何在PHP开发中应用这些设计模式,从而写出更加高效、灵活和易于维护的代码。
本文探讨了PHP中常用的设计模式及其在实际项目中的应用。内容涵盖设计模式的基本概念、分类和具体使用场景,重点介绍了单例模式、工厂模式和观察者模式等常见模式。通过具体的代码示例,展示了如何在PHP项目中有效利用设计模式来提升代码的可维护性和扩展性。文章还讨论了设计模式的选择原则和注意事项,帮助开发者在不同情境下做出最佳决策。