享元模式
一、 概念
二、享元的用途
三、结构和分类
1、单纯享元模式 
2、复合享元模式
四、享元模式的优缺点


一、概念

 Flyweight在拳击比赛中指最轻量级,即“蝇量级”或“雨量级”,这里选择使用“享元模式”的意译,是因为这样更能反映模式的用意。享元模式是对象的结构模式。享元模式以共享的方式高效地支持大量的细粒度对象。
也就是说在一个系统中如果有多个相同的对象,那么只共享一份就可以了,不必每个都去实例化一个对象。


二、享元的用途

在java应用中,会出现许多String a=”123”,String b=”123”之类的String类型的变量,如果只是小应用,到还好,假设是一个庞大的系统,有好多处都需要用定义String a=”223”,那开销可想而知,而JDK的开发者自然想到了这点,采用了享元模式解决创建大量相同String变量带来的开销问题

在JAVA语言中,String类型就是使用了享元模式。String对象是final类型,对象一旦创建就不可改变。在JAVA中字符串常量 都是存在常量池中的,JAVA会确保一个字符串常量在常量池中只有一个拷贝。String a=”abc”,其中”abc”就是一个字符串常量。

public class Test {public static void main(String[] args) {String a = "abc";String b = "abc";System.out.println(a==b);}
}

  上面的例子中结果为:true ,这就说明a和b两个引用都指向了常量池中的同一个字符串常量”abc”。这样的设计避免了在创建N多相同对象时所产生的不必要的大量的资源消耗。


三、结构和分类
享元模式采用一个共享来避免大量拥有相同内容对象的开销。这种开销最常见、最直观的就是内存的损耗。享元对象能做到共享的关键是区分内蕴状态(Internal State)和外蕴状态(ExternalState)。

一个内蕴状态是存储在享元对象内部的,并且是不会随环境的改变而有所不同。因此,一个享元可以具有内蕴状态并可以共享。

一个外蕴状态是随环境的改变而改变的、不可以共享的。享元对象的外蕴状态必须由客户端保存,并在享元对象被创建之后,在需要使用的时候再传入到享元对象内部。外蕴状态不可以影响享元对象的内蕴状态,它们是相互独立的。

享元模式可以分成单纯享元模式和复合享元模式两种形式。
1、单纯享元模式  
在单纯的享元模式中,所有的享元对象都是可以共享的。

角色如下:

●抽象享元(Flyweight)角色 :给出一个抽象接口,以规定出所有具体享元角色需要实现的方法。

●具体享元(ConcreteFlyweight)角色:实现抽象享元角色所规定出的接口。如果有内蕴状态的话,必须负责为内蕴状态提供存储空间。

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

源代码:
(下面的例子外蕴状态可以看做一个点菜的人 内蕴状态可以看做一个菜名,点菜的人一般经常变 但是菜的种类是固定的)

抽象享元角色类

//抽象享元角色类
public interface Flyweight {//一个示意性方法,参数state是外蕴状态public void operation(String state);
}

 具体享元角色类ConcreteFlyweight有一个内蕴状态,在本例中一个Character类型的intrinsicState属性代表,它的值应当在享元对象被创建时赋予。所有的内蕴状态在对象创建之后,就不会再改变了。
如果一个享元对象有外蕴状态的话,所有的外部状态都必须存储在客户端,在使用享元对象时,再由客户端传入享元对象。这里只有一个外蕴状态,operation()方法的参数state就是由外部传入的外蕴状态。

//具体享元角色类
public class ConcreteFlyweight implements Flyweight {private String intrinsicState = null;//内蕴状态/*** 构造函数,内蕴状态作为参数传入* * @param intrinsicState*/public ConcreteFlyweight(String intrinsicState) {this.intrinsicState = intrinsicState;}/*** 外蕴状态作为参数传入方法中,改变方法的行为, 但是并不改变对象的内蕴状态。*/@Overridepublic void operation(String state) {System.out.println("内蕴状态= " + this.intrinsicState);System.out.println("外蕴状态 = " + state);}}

享元工厂角色类,必须指出的是,客户端不可以直接将具体享元类实例化,而必须通过一个工厂对象,利用一个factory()方法得到享元对象。一般而言,享元工厂对象在整个系统中只有一个,因此也可以使用单例模式。
当客户端需要单纯享元对象的时候,需要调用享元工厂的factory()方法,并传入所需的单纯享元对象的内蕴状态,由工厂方法产生所需要的享元对象。

//享元工厂角色类
public class FlyweightFactory {// 一个用来存所有享元对象的集合 String表示对象的键的类型 ->内蕴状态 ;Flyweight表示对象值的类型private Map<String, Flyweight> files = new HashMap<String, Flyweight>();public Flyweight factory(String intrinsicState) {// 先从缓存中查找对象Flyweight fly = files.get(intrinsicState);if (fly == null) {// 如果对象不存在则创建一个新的Flyweight对象fly = new ConcreteFlyweight(intrinsicState);// 把这个新的Flyweight对象添加到缓存中files.put(intrinsicState, fly);}return fly;}//得到存对象的集合的长度public int getFlyWeightSize() {return files.size();}}

public class Client {public static void main(String[] args) {FlyweightFactory factory = new FlyweightFactory();Flyweight fly1 = factory.factory(new String("辣椒炒肉"));fly1.operation("汤高点菜");Flyweight fly2 = factory.factory(new String("牛肉"));fly2.operation("周思远点菜");Flyweight fly3 = factory.factory(new String("辣椒炒肉"));fly3.operation("汤高点菜");System.out.println(fly1==fly3);System.out.println("被点不同的菜的个数"+factory.getFlyWeightSize());}}

结果:
内蕴状态= 辣椒炒肉
外蕴状态 = 汤高点菜
内蕴状态= 牛肉
外蕴状态 = 周思远点菜
内蕴状态= 辣椒炒肉
外蕴状态 = 汤高点菜
true
被点不同的菜的个数2

虽然客户端申请了三个享元对象,但是实际创建的享元对象只有两个,这就是共享的含义


2、复合享元模式
  在单纯享元模式中,所有的享元对象都是单纯享元对象,也就是说都是可以直接共享的。还有一种较为复杂的情况,将一些单纯享元使用合成模式加以复合,形成复合享元对象。这样的复合享元对象本身不能共享,但是它们可以分解成单纯享元对象,而后者则可以共享。


  
  
角色如下:
●抽象享元(Flyweight)角色 :给出一个抽象接口,以规定出所有具体享元角色需要实现的方法。
  
●复合享元(ConcreteCompositeFlyweight)角色 :复合享元角色所代表的对象是不可以共享的,但是一个复合享元对象可以分解成为多个本身是单纯享元对象的组合。复合享元角色又称作不可共享的享元对象。
  
●享元工厂(FlyweightFactory)角色 :本角 色负责创建和管理享元角色。本角色必须保证享元对象可以被系统适当地共享。当一个客户端对象调用一个享元对象的时候,享元工厂角色会检查系统中是否已经有 一个符合要求的享元对象。如果已经有了,享元工厂角色就应当提供这个已有的享元对象;如果系统中没有一个适当的享元对象的话,享元工厂角色就应当创建一个 合适的享元对象。
(下面的例子外蕴状态可以看做一个点菜的人 内蕴状态可以看做一个菜名,点菜的人一般经常变 但是菜的种类是固定的)


源代码
抽象享元角色类

//抽象享元角色类
public interface Flyweight {//一个示意性方法,参数state是外蕴状态public void operation(String state);
}

 
具体享元角色类

///具体享元角色类
public class ConcreteFlyweight implements Flyweight {private String intrinsicState = null;//内蕴状态/*** 构造函数,内蕴状态作为参数传入* * @param intrinsicState2*/public ConcreteFlyweight(String intrinsicState2) {this.intrinsicState = intrinsicState2;}/*** 外蕴状态作为参数传入方法中,改变方法的行为, 但是并不改变对象的内蕴状态。*/@Overridepublic void operation(String state) {System.out.println("内蕴状态= " + this.intrinsicState);System.out.println("外蕴状态 = " + state);}}

复合享元对象是由单纯享元对象通过复合而成的,因此它提供了add()这样的聚集管理方法。由于一个复合享元对象具有不同的聚集元素,这些聚集元素在复合享元对象被创建之后加入,这本身就意味着复合享元对象的状态是会改变的,因此复合享元对象是不能共享的。
  复合享元角色实现了抽象享元角色所规定的接口,也就是operation()方法,这个方法有一个参数,代表复合享元对象的外蕴状态。一个复合 享元对象的所有单纯享元对象元素的外蕴状态都是与复合享元对象的外蕴状态相等的;而一个复合享元对象所含有的单纯享元对象的内蕴状态一般是不相等的,不然 就没有使用价值了。
  

//复合享元角色
public class ConcreteCompositeFlyweight implements Flyweight {private Map<String, Flyweight> files = new HashMap<String, Flyweight>();/*** 增加一个新的单纯享元对象到聚集中*/public void add(String intrinsicState, Flyweight fly) {files.put(intrinsicState, fly);}/*** 外蕴状态作为参数传入到方法中*/@Overridepublic void operation(String state) {Flyweight fly = null;for (String intrinsicState : files.keySet()) {fly = files.get(intrinsicState);fly.operation(state);}}}

享元工厂角色提供两种不同的方法,一种用于提供单纯享元对象,另一种用于提供复合享元对象。

//享元工厂角色类
public class FlyweightFactory {// 一个用来存所有享元对象的集合 String表示对象的键的类型 ->内蕴状态 ;Flyweight表示对象值的类型private Map<String, Flyweight> files = new HashMap<String, Flyweight>();/*** 复合享元工厂方法*/public Flyweight factory(List<String> compositeState) {ConcreteCompositeFlyweight compositeFly = new ConcreteCompositeFlyweight();for (String intrinsicState : compositeState) {compositeFly.add(intrinsicState, this.factory(intrinsicState));}return compositeFly;}/*** 单纯享元工厂方法*/public Flyweight factory(String intrinsicState) {// 先从缓存中查找对象Flyweight fly = files.get(intrinsicState);if (fly == null) {// 如果对象不存在则创建一个新的Flyweight对象fly = new ConcreteFlyweight(intrinsicState);// 把这个新的Flyweight对象添加到缓存中files.put(intrinsicState, fly);}return fly;}// 得到存对象的集合的长度public int getFlyWeightSize() {return files.size();}}

public class Client {public static void main(String[] args) {List<String> compositeState = new ArrayList<String>();compositeState.add("辣椒炒肉");compositeState.add("牛肉");compositeState.add("鸡肉");compositeState.add("辣椒炒肉");compositeState.add("牛肉");FlyweightFactory flyFactory = new FlyweightFactory();Flyweight compositeFly1 = flyFactory.factory(compositeState);Flyweight compositeFly2 = flyFactory.factory(compositeState);compositeFly1.operation("汤高点菜");//外蕴状态是同一个System.out.println();compositeFly2.operation("周思远点菜");System.out.println("---------------------------------");System.out.println("复合享元模式是否可以共享对象:" + (compositeFly1 == compositeFly2));String state = "牛肉";Flyweight fly1 = flyFactory.factory(state);Flyweight fly2 = flyFactory.factory(state);System.out.println("单纯享元模式是否可以共享对象:" + (fly1 == fly2));}}

结果:
内蕴状态= 辣椒炒肉
外蕴状态 = 汤高点菜
内蕴状态= 鸡肉
外蕴状态 = 汤高点菜
内蕴状态= 牛肉
外蕴状态 = 汤高点菜

内蕴状态= 辣椒炒肉
外蕴状态 = 周思远点菜
内蕴状态= 鸡肉
外蕴状态 = 周思远点菜
内蕴状态= 牛肉
外蕴状态 = 周思远点菜

复合享元模式是否可以共享对象:false
单纯享元模式是否可以共享对象:true

从运行结果可以看出,一个复合享元对象的所有单纯享元对象元素的外蕴状态都是与复合享元对象的外蕴状态相等的。即外运状态都等于汤高点菜或者周思远点菜。

从运行结果可以看出,一个复合享元对象所含有的单纯享元对象的内蕴状态一般是不相等的。即内蕴状态分别为各自的菜名。

从运行结果可以看出,复合享元对象是不能共享的。即使用相同的对象compositeState通过工厂分别两次创建出的对象不是同一个对象。

从运行结果可以看出,单纯享元对象是可以共享的。即使用相同的对象state通过工厂分别两次创建出的对象是同一个对象。


四、享元模式的优缺点

  享元模式的优点在于它大幅度地降低内存中对象的数量。但是,它做到这一点所付出的代价也是很高的:
  
  ●  享元模式使得系统更加复杂。为了使对象可以共享,需要将一些状态外部化,这使得程序的逻辑复杂化。
  ●  享元模式将享元对象的状态外部化,而读取外部状态使得运行时间稍微变长。

以上内容来自平时所看书籍和网络资源整理测试所得,如有不完善之处,欢迎指正!

Java设计模式(十二)----享元模式相关推荐

  1. GoF设计模式(十二) - 享元模式

    前言 享元模式(Flyweight Pattern)运用共享技术有效地支持大量细粒度的对象,以减少内存占用和提高性能.这种类型的设计模式属于结构型模式,它提供了减少对象数量从而改善应用所需的对象结构的 ...

  2. 11.java设计模式(读书笔记)享元模式

    参考:http://blog.csdn.net/jason0539/article/details/22908915 解释一下概念:也就是说在一个系统中如果有多个相同的对象,那么只共享一份就可以了,不 ...

  3. 设计模式(十)享元模式Flyweight(结构型)

    设计模式(十)享元模式Flyweight(结构型) 说明: 相对于其它模式,Flyweight模式在PHP实现似乎没有太大的意义,因为PHP的生命周期就在一个请求,请求执行完了,php占用的资源都被释 ...

  4. 设计模式十二之组合模式

    设计模式十二之组合模式 1. 模式的定义与特点 1.1 模式的定义 1.2 模式的特点 1.3 模式的使用场景 2. 模式的结构与实现 2.1 模式的结构 2.2 模式的实现 3. 模式在开源软件中的 ...

  5. 结构型设计模式(五) 之 享元模式是池技术的表现

    1 定义 享元模式(Flyweight Pattern)属于结构型设计模式之一,它主要是使用共享对象有效地避免了创建过多的性质相近的对象,从而降低内存的占用,提高程序的性能.它也是池技术的重要实现方式 ...

  6. Java描述设计模式(18):享元模式

    本文源码:GitHub·点这里 || GitEE·点这里 一.使用场景 应用代码 public class C01_InScene {public static void main(String[] ...

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

    一.引言 今天我们要讲[结构型]设计模式的第六个模式,该模式是[享元模式],英文名称是:Flyweight Pattern.还是老套路,先从名字上来看看."享元"是不是可以这样理解 ...

  8. 【设计模式自习室】享元模式 Flyweight Pattern:减少对象数量

    前言 <设计模式自习室>系列,顾名思义,本系列文章带你温习常见的设计模式.主要内容有: 该模式的介绍,包括: 引子.意图(大白话解释) 类图.时序图(理论规范) 该模式的代码示例:熟悉该模 ...

  9. java23设计模式---class10、享元模式(FlyWeight)

    文章目录 一.基本介绍 1.定义 2.优点 3.缺点 4.角色 1)抽象享元类 2)具体享元类 3)享元工厂类 4)组合享元类 5.内部状态和外部状态 二.应用情景 1.线程池 2.String 3. ...

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

    一.享元模式简介(Brief Introduction) 享元模式(Flyweight Pattern),运用共享技术有效支持大量细粒度的对象. Use sharing to support larg ...

最新文章

  1. css3之transition、transform、animation比较
  2. Android读取短信和联系人
  3. 安装cadence ic5141时碰到字体问题
  4. 用python验证猜想之类的例子_python练手好例子:验证哥德巴赫猜想
  5. 安装PHP5,安装PHP7
  6. servlet的应用------request对象和bean实体的反射关系
  7. Phoenix Contact 多款工业产品被曝多个高危漏洞
  8. 基于OpenCV的计算机视觉入门(1)图片操作入门
  9. 2014年工作中遇到的20个问题:161-180
  10. Python3——通用序列操作
  11. TextCNN文本分类实践
  12. VS2013 ADO.NET 连接 SQLEXPRESS
  13. matlab环境下图像分形维数的计算,MATLAB环境下图像分形维数的计算_杨书申
  14. Excel数据导入___你hold住么(二)
  15. “WebProxyWrapper”的对象强制转换为类型“System.Net.WebProxy
  16. Android学习日记(yzy):Notification的简单运用
  17. RadioGroup+Fragment+PulltorefreshScrollView+GridView+侧拉菜单+TabLayout+ViewPager
  18. 招聘方眼里的猎聘和Boss直聘直观对比
  19. cad2016的自动修复此计算机,CAD中遇到文件损坏,别着急,这几招能帮你挽回损失...
  20. 安装matlab贝叶斯网络工具箱

热门文章

  1. ie6-7中会出现图片下有空隙
  2. 艾伟_转载:VS2008连接SQL Server数据库文件出错的解决方案
  3. 祝各位朋友元旦快乐,新年大发!!!
  4. 推荐几个.NET开源图表组件
  5. netlink 0005 -- Generic Netlink详解
  6. Audio播放流程(三)---NuPlayer流程之setAudioStreamType以及prepare
  7. 用glew,glfw,FreeImag实现opengl画图-第五课 摄像机
  8. L2-010 排座位 (25 分)
  9. python web框架 多线程_Django基础一之web框架的本质
  10. mapreduce多路输出实例