享元模式 - 设计模式学习

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

  下面给出享元模式的结构图:

      

  下面给出享元模式的基本代码结构:

namespace ConsoleApplication1
{
    //Flyweight类,它是所有具体享元类的超类或接口,通过这个接口,Flyweight可以接受并作用于外部状态
    abstract class Flyweight
    {
        public abstract void Operation(int extrinsicstate);
    }

    //ConcreteFlyweight是继承Flyweight或实现Flyweight接口,并为内部状态增加存储空间
    class ConcreteFlyweight : Flyweight
    {
        public override void Operation(int extrinsicstate)
        {
            Console.WriteLine("具体Flyweight:" + extrinsicstate);
        }
    }

    //UnsharedConcreteFlyweight是指那些不需要共享的Flyweight子类。因为Flyweight接口共享称为可能,但它并不强制共享
    class UnsharedConcreteFlyweight : Flyweight
    {
        public override void Operation(int extrinsicstate)
        {
            Console.WriteLine("不共享的具体Flyweight:" + extrinsicstate);
        }
    }

    //FlyweightFactory,是一个享元工厂,用来创建并管理Flyweight对象。它主要是用来确保合理地共享Flyweight
    //当用户请求一个Flyweight时,FlyweightFactory对象提供一个已创建的实例,如不存在则创建一个。
    class FlyweightFactory
    {
        private Hashtable flyweights = new Hashtable();

        public FlyweightFactory()   //初始化工厂,先生成3个实例
        {
            flyweights.Add("X", new ConcreteFlyweight());
            flyweights.Add("Y", new ConcreteFlyweight());
            flyweights.Add("Z", new ConcreteFlyweight());
        }

        public Flyweight GetFlyweight(string key)   //根据客户端请求,获得已生成实例
        {
            return ((Flyweight)flyweights[key]);
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            int extrinsicstate = 22;    //代码外部状态

            FlyweightFactory f = new FlyweightFactory();

            Flyweight fx = f.GetFlyweight("X");
            fx.Operation(--extrinsicstate);

            Flyweight fy = f.GetFlyweight("Y");
            fx.Operation(--extrinsicstate);

            Flyweight fz = f.GetFlyweight("Z");
            fx.Operation(--extrinsicstate);

            Flyweight uf = new UnsharedConcreteFlyweight();
            uf.Operation(--extrinsicstate);

            Console.ReadKey();
        }
    }
}

  结果如图所示:

  

  UnsharedConcreteFlyweight用于解决那些不需要共享对象的问题。因为享元模式在使用的时候,既有需要共享的部分也有不需要共享的部分,此时UnsharedConcreteFlyweight的作用就出来了。

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

  在现实中什么时候才应该考虑使用享元模式呢?

  如果一个应用程序使用了大量的对象,而大量的这些对象造成了很大的存储开销时就应该考虑使用;还有就是对象的大多数状态可以外部状态,如果删除对象的外部状态,那么可以用相对较少的共享对象取代很多组对象,此时可以考虑使用享元模式。

  下面回到《大话设计模式》里面的网站建设的例子:

namespace ConsoleApplication1
{
    //用户
    public class User
    {
        private string name;
        public User(string name)
        {
            this.name = name;
        }

        public string Name
        {
            get { return name; }
        }
    }

    //网站抽象类
    abstract class WebSite
    {
        public abstract void Use(User user);    //"使用"方法需要传递"用户"对象
    }

    //具体网站类
    class ConcreteWebSite : WebSite
    {
        private string name = "";

        public ConcreteWebSite(string name)
        {
            this.name = name;
        }

        public override void Use(User user)     //实现"Use"方法
        {
            Console.WriteLine("网站分类:" + name + " 用户:" + user.Name);
        }
    }

    //网站工厂类
    class WebSiteFactory
    {
        private Hashtable flyweights = new Hashtable();

        //获得网站分类
        public WebSite GetWebSiteCategory(string key)
        {
            if (!flyweights.ContainsKey(key))
            {
                flyweights.Add(key, new ConcreteWebSite(key));
            }
            return ((WebSite)flyweights[key]);
        }

        //获得网站分类总数
        public int GetWebSiteCount()
        {
            return flyweights.Count;
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            WebSiteFactory f = new WebSiteFactory();

            WebSite ws1 = f.GetWebSiteCategory("产品展示");
            ws1.Use(new User("小菜"));

            WebSite ws2 = f.GetWebSiteCategory("产品展示");
            ws1.Use(new User("大鸟"));

            WebSite ws3 = f.GetWebSiteCategory("产品展示");
            ws1.Use(new User("娇娇"));

            WebSite ws4 = f.GetWebSiteCategory("博客");
            ws1.Use(new User("关羽"));

            WebSite ws5 = f.GetWebSiteCategory("博客");
            ws1.Use(new User("张飞"));

            WebSite ws6 = f.GetWebSiteCategory("博客");
            ws1.Use(new User("赵云"));

            Console.WriteLine("得到网站分类的总数为{0}",f.GetWebSiteCount());

            Console.ReadKey();
        }
    }
}

  结果如下图所示:

    

时间: 2024-08-09 17:11:09

享元模式 - 设计模式学习的相关文章

不一样的享元模式(设计模式四)

前言 享元模式,表面意思是共享单元,属于结构型设计模式.哦?good啊,如今共享文化高大上,共享肯定节约很多资源吧,肯定用的地方挺多吧,然而并不多,但是又是不可或缺的一种模式. 至于为什么,请看正文部分,将会通过计算分析出为什么用的地方不多,或者说有些地方为什么不该用,同时得出为什么属于结构型,到底属于结构型的哪一种. 开车触发 介绍一下什么是结构型: 结构型模式所描述的是如何将类和对象结合在一起来形成一个更大的结构,它描述两种不同的事物:类和对象,根据这一点,可分为类结构型和对象结构型模式.类

三分钟理解“享元模式”——设计模式轻松掌握

享原模式的官方定义: 运用共享技术有效地支持大量细粒度的对象. 大白话讲: 一个类它可能生成好多对象,但这些对象根据属性值的不同一共分成N类,每种类型中属性值都是一样的.在这种情况下,如果创建好多对象,那么这些对象中很多属性值都是重复的,从而造成了大量的内存浪费. 而享元模式能够解决重复对象的内存浪费的问题. 享元模式使用一个工厂类,在工厂类中为每种类型创建一个对象,而且每种类型的对象只有一个. 当客户端需要某种类型的对象的时候,工厂将已创建好的对象给客户端.由于不创建新的对象了,所以节省了内存

【转】享元模式——实现对象的复用

[作者:刘伟  http://blog.csdn.net/lovelion] 当前咱们国家正在大力倡导构建和谐社会,其中一个很重要的组成部分就是建设资源节约型社会,“浪费可耻,节俭光荣”.在软件系统中,有时候也会存在资源浪费的情况,例如在计算机内存中存储了多个完全相同或者非常相似的对象,如果这些对象的数量太多将导致系统运行代价过高,内存属于计算机的“稀缺资源”,不应该用来“随便浪费”,那么是否存在一种技术可以用于节约内存使用空间,实现对这些相同或者相似对象的共享访问呢?答案是肯定,这种技术就是我

java/android 设计模式学习笔记(13)---享元模式

这篇我们来介绍一下享元模式(Flyweight Pattern),Flyweight 代表轻量级的意思,享元模式是对象池的一种实现.享元模式用来尽可能减少内存使用量,它适合用于可能存在大量重复对象的场景,缓存可共享的对象,来达到对象共享和避免创建过多对象的效果,这样一来就可以提升性能,避免内存移除和频繁 GC 等. 享元模式的一个经典使用案例是文本系统中图形显示所用的数据结构,一个文本系统能够显示的字符种类就是那么几十上百个,那么就定义这么些基础字符对象,存储每个字符的显示外形和其他的格式化数据

设计模式学习08:享元模式

设计模式学习08:享元模式 参考资料 https://www.jianshu.com/p/a2aa75939766 https://www.cnblogs.com/adamjwh/p/9070107.html 简介 Use sharing to support large numbers of fine-grained objects efficiently. 使用共享对象可有效地支持大量的细粒度的对象. 享元模式(Flyweight)又称为 轻量级模式,它是一种对象结构型模式. 面向对象技术可

javascript设计模式学习之十一——享元模式

一.享元模式的定义及使用场景 享元模式是一种用于性能优化的模式,如果系统中因为创建了大量类似对象而导致内存占用过高,享元模式就非常有用了.享元模式的核心是运用共享技术来有效支持大量细粒度的对象. 享元模式的关键是区分内部状态和外部状态,剥离了外部状态的对象成为共享对象,外部状态在必要时被传入共享对象来组装成一个完整的对象.那些可以被对象共享的属性通常就被划分为内部状态.

《大话设计模式》学习笔记22:享元模式

网站共享代码示例: 1.Flyweight: public abstract class WebSite { public abstract void Use(); } 2.ConcreteeFlyweight: public class ConcreteWebSite:WebSite { private string webSiteCategory; public ConcreteWebSite(string webSiteCategory) { this.webSiteCategory =

C#设计模式-享元模式

前言 最近开始花点心思研究下设计模式,主要还是让自己写的代码可重用性高.保证代码可靠性.所谓设计模式,我找了下定义:是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.毫无疑问,设计模式于己于他人于系统都是多赢的:设计模式使代码编制真正工程化:设计模式是软件工程的基石脉络,如同大厦的结构一样. 为什么要提倡“Design Pattern(设计模式)”? 根本原因是为了代码复用,增加可维护性.因此这次我们来学习下设计模式,最后会通过C#语言来实现这些设计模式作为例子,深刻理解其中的

设计模式(十二): Flyweight享元模式 -- 结构型模式

说明: 相对于其它模式,Flyweight模式在PHP实现似乎没有太大的意义,因为PHP的生命周期就在一个请求,请求执行完了,php占用的资源都被释放.我们只是为了学习而简单做了介绍. 1. 概述 面向对象技术可以很好地解决系统一些灵活性或可扩展性或抽象性的问题,但在很多情况下需要在系统中增加类和对象的个数.当对象数量太多时,将导致运行代价过高,带来性能下降等问题.比如:例子1:图形应用中的图元等对象.字处理应用中的字符对象等. 2.解决方案: 享元模式(Flyweight):对象结构型模式运用