设计模式(十)享元模式Flyweight(结构型)
说明:
相对于其它模式,Flyweight模式在PHP实现似乎没有太大的意义,因为PHP的生命周期就在一个请求,请求执行完了,php占用的资源都被释放。我们只是为了学习而简单做了介绍。

1. 概述

面向对象技术可以很好地解决系统一些灵活性或可扩展性或抽象性的问题,但在很多情况下需要在系统中增加类和对象的个数。当对象数量太多时,将导致运行代价过高,带来性能下降等问题。比如:
例子1:图形应用中的图元等对象、字处理应用中的字符对象等。

2.解决方案:

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

它使用共享物件,用来尽可能减少内存使用量以及分享资讯给尽可能多的相似物件;它适合用于当大量物件只是重复因而导致无法令人接受的使用大量内存。通常物件中的部分状态是可以分享。常见做法是把它们放在外部数据结构,当需要使用时再将它们传递给享元。

4. 适用性

1)一个应用程序使用大量相同或者相似的对象,造成很大的存储开销。

2)对象的大部分状态都可以外部化,可以将这些外部状态传入对象中。

3)如果删除对象的外部状态,那么可以用相对较少的共享对象取代很多组对象。

4) 应用程序不依赖于对象标识。由于Flyweight对象可以被共享,对于概念上明显有别的对象,标识测试将返回真值。

5)使用享元模式需要维护一个存储享元对象的享元池,而这需要耗费资源,因此,应当在多次重复使用享元对象时才值得使用享元模式

5.结构

6.构建模式的组成

1) 抽象享元类(Flyweight): 描述一个接口,通过这个接口flyweight可以接受并作用于外部状态。
2) 具体享元类(ConcreteFlyweight):实现Flyweight接口 ,并为内部状态( 如果有的话 )增加存储空间 。ConcreteFlyweight对象必须是可共享的。它所存储的状态必须是内部的;即,它必须独立于ConcreteFlyweight对象的场景。
3) 非共享具体享元类(UnsharedConcreteFlyweight):— 并非所有的Flyweight子类都需要被共享。Flyweight接口使共享成为可能,但它并不强制共享。在Flyweight对象结构的某些层次,UnsharedConcreteFlyweight对象通常将ConcreteFlyweight对象作为子节点。
4) 享元工厂类(FlyweightFactory):创建并管理flyweight对象, 确保合理地共享flyweight。本角色必须保证享元对象可以被系统适当地共享。当一个客户端对象调用一个享元对象   flyweight的时候,享元工厂角色(Flyweight Factory对象)会检查系统中是否已经有一个符合要求的享元对象。如果已经有了,享元工厂角色就应当提供这个已有的享元对象;如果系统中没有一个适当的享元对象的话,享元工厂角色就应当创建一个合适的享元对象。
5)客户(Client):维持一个对flyweight的引用。计算或存储一个(多个)flyweight的外部状态。

7. 效果

享元模式的优点:
1)享元模式的优点在于它可以极大减少内存中对象的数量,使得相同对象或相似对象在内存中只保存一份。
2)享元模式的外部状态相对独立,而且不会影响其内部状态,从而使得享元对象可以在不同的环境中被共享。
享元模式的缺点:
1)享元模式使得系统更加复杂,需要分离出内部状态和外部状态,这使得程序的逻辑复杂化。
2)为了使对象可以共享,享元模式需要将享元对象的状态外部化,而读取外部状态使得运行时间变长。

8.实现

享元模式可以分成单纯享元模式和复合享元模式两种形式。

【单纯享元模式】

  在单纯的享元模式中,所有的享元对象都是可以共享的。

源代码:

[php] view plaincopy print?
  1. <?php
  2. /**
  3. * 单纯享元模式 
  4. * @author guisu
  5. */
  6. /**
  7. * 抽象享元角色
  8. */
  9. abstract class Flyweight {
  10. /**
  11. * 示意性方法
  12. * @param string $state 外部状态
  13. */
  14. abstract public function operation($state);
  15. }
  16. /**
  17. * 具体享元角色
  18. */
  19. class ConcreteFlyweight extends Flyweight {
  20. private $_intrinsicState = null;
  21. /**
  22. * 构造方法
  23. * @param string $state  内部状态
  24. */
  25. public function __construct($state) {
  26. $this->_intrinsicState = $state;
  27. }
  28. public function operation($state) {
  29. echo 'ConcreteFlyweight operation, Intrinsic State = ' . $this->_intrinsicState
  30. . ' Extrinsic State = ' . $state . '<br />';
  31. }
  32. }
  33. /**
  34. * 享元工厂角色
  35. */
  36. class FlyweightFactory {
  37. private $_flyweights;
  38. public function __construct() {
  39. $this->_flyweights = array();
  40. }
  41. public function getFlyweigth($state) {
  42. if (isset($this->_flyweights[$state])) {
  43. return $this->_flyweights[$state];
  44. } else {
  45. return $this->_flyweights[$state] = new ConcreteFlyweight($state);
  46. }
  47. }
  48. }
  49. class Client{
  50. static function main (){
  51. $flyweightFactory = new FlyweightFactory();
  52. $flyweight = $flyweightFactory->getFlyweigth('state A');
  53. $flyweight->operation('other state A');
  54. $flyweight = $flyweightFactory->getFlyweigth('state B');
  55. $flyweight->operation('other state B');
  56. }
  57. }
  58. ?>

【复合享元模式】
复合享元模式对象是由一些单纯享元使用合成模式加以复合而成
复合享元角色所代表的对象是不可以共享的,但是一个复合享元对象可以分解成为多个本身是单纯享元对象的组合。

[php] view plaincopy print?
  1. <?php
  2. /**
  3. * 复合享元模式
  4. *
  5. */
  6. /**
  7. * 抽象享元角色
  8. */
  9. abstract class Flyweight {
  10. /**
  11. * 示意性方法
  12. * @param string $state 外部状态
  13. */
  14. abstract public function operation($state);
  15. }
  16. /**
  17. * 具体享元角色
  18. */
  19. class ConcreteFlyweight extends Flyweight {
  20. private $_intrinsicState = null;
  21. /**
  22. * 构造方法
  23. * @param string $state  内部状态
  24. */
  25. public function __construct($state) {
  26. $this->_intrinsicState = $state;
  27. }
  28. public function operation($state) {
  29. echo 'ConcreteFlyweight operation, Intrinsic State = ' . $this->_intrinsicState
  30. . ' Extrinsic State = ' . $state . '<br />';
  31. }
  32. }
  33. /**
  34. * 不共享的具体享元,客户端直接调用
  35. */
  36. class UnsharedConcreteFlyweight extends Flyweight {
  37. private $_flyweights;
  38. /**
  39. * 构造方法
  40. * @param string $state  内部状态
  41. */
  42. public function __construct() {
  43. $this->_flyweights = array();
  44. }
  45. public function operation($state) {
  46. foreach ($this->_flyweights as $flyweight) {
  47. $flyweight->operation($state);
  48. }
  49. }
  50. public function add($state, Flyweight $flyweight) {
  51. $this->_flyweights[$state] = $flyweight;
  52. }
  53. }
  54. /**
  55. * 享元工厂角色
  56. */
  57. class FlyweightFactory {
  58. private $_flyweights;
  59. public function __construct() {
  60. $this->_flyweights = array();
  61. }
  62. public function getFlyweigth($state) {
  63. if (is_array($state)) { //  复合模式
  64. $uFlyweight = new UnsharedConcreteFlyweight();
  65. foreach ($state as $row) {
  66. $uFlyweight->add($row, $this->getFlyweigth($row));
  67. }
  68. return $uFlyweight;
  69. } else if (is_string($state)) {
  70. if (isset($this->_flyweights[$state])) {
  71. return $this->_flyweights[$state];
  72. } else {
  73. return $this->_flyweights[$state] = new ConcreteFlyweight($state);
  74. }
  75. } else {
  76. return null;
  77. }
  78. }
  79. }
  80. class Client{
  81. static function main (){
  82. $flyweightFactory = new FlyweightFactory();
  83. $flyweight = $flyweightFactory->getFlyweigth('state A');
  84. $flyweight->operation('other state A');
  85. $flyweight = $flyweightFactory->getFlyweigth('state B');
  86. $flyweight->operation('other state B');
  87. /* 复合对象*/
  88. $uflyweight = $flyweightFactory->getFlyweigth(array('state A', 'state B'));
  89. $uflyweight->operation('other state A');
  90. }
  91. }
  92. ?>

9. 与其他相关模式

客户端要引用享元对象,是通过工厂对象创建或者获得的,客户端每次引用一个享元对象,都是可以通过同一个工厂对象来引用所需要的享元对象。因此,可以将享元工厂设计成单例模式,这样就可以保证客户端只引用一个工厂实例。因为所有的享元对象都是由一个工厂对象统一管理的,所以在客户端没有必要引用多个工厂对象。不管是单纯享元模式还是复合享元模式中的享元工厂角色,都可以设计成为单例模式,对于结果是不会有任何影响的。

Composite模式:Flyweight模式通常和Composite 模式结合起来,用共享叶结点的有向无环图实现一个逻辑上的层次结构。复合享元模式实际上是单纯享元模式与合成模式的组合。单纯享元对象可以作为树叶对象来讲,是可以共享的,而复合享元对象可以作为树枝对象, 因此在复合享元角色中可以添加聚集管理方法。通常,最好用Flyweight实现State 和Strategy 对象。

10.总结

1)  享元模式是一个考虑系统性能的设计模式,通过使用享元模式可以节约内存空间,提高系统的性能。 
2)  享元模式的核心在于享元工厂类,享元工厂类的作用在于提供一个用于存储享元对象的享元池,用户需要对象时,首先从享元池中获取,如果享 元池中不存在,则创       建一个新的享元对象返回给用户,并在享元池中保存该新增对象。
3)   享元模式以共享的方式高效地支持大量的细粒度对象,享元对象能做到共享的关键是区分内部状态(Internal State)和外部状态(External State)。
(1)   内部状态是存储在享元对象内部并且不会随环境改变而改变的状态,因此内部状态可以共享。
(2)   外部状态是随环境改变而改变的、不可以共享的状态。享元对象的外部状态必须由客户端保存,并在享元对象被创建之后,在需要使用的时候 再传入到享元对               象内部。一个外部状态与另一个外部状态之间是相互独立的。

设计模式(十)享元模式Flyweight(结构型)相关推荐

  1. 享元模式 FlyWeight 结构型 设计模式(十五)

    享元模式(FlyWeight)  "享"取"共享"之意,"元"取"单元"之意. 意图 运用共享技术,有效的支持大量细粒度 ...

  2. 享元模式(结构型模式)

    7.享元模式 ​ 享元模式(Flyweight Pattern)运用共享技术来有效地支持大量细粒度对象的复用.它通过共享已经存在的对象来大幅度减少需要创建的对象数量.避免大量相似对象的开销,从而提高系 ...

  3. java设计模式之——外观模式、组合模式、装饰模式、享元模式(结构型)

    一.外观模式 定义:为子系统中一组接口提供一致的界面,此模式定义了一个高层接口,该接口使得子系统更加容易使用. 使用场景:一.在软件设计初级阶段,应该有意识的将软件两个层分开,比如经典的三层架构,降低 ...

  4. 设计模式之享元模式(Flyweight)摘录

    23种GOF设计模式一般分为三大类:创建型模式.结构型模式.行为模式. 创建型模式抽象了实例化过程,它们帮助一个系统独立于如何创建.组合和表示它的那些对象.一个类创建型模式使用继承改变被实例化的类,而 ...

  5. 设计模式(18):结构型-享元模式(Flyweight)

    设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了可重用代码.让代码更容易被他人理解.保证代码可靠性. 毫无疑问,设计模式于 ...

  6. Flyweight(享元)--对象结构型模式

    Flyweight(享元)–对象结构型模式 一.意图 运行共享技术有效地支持大量细粒度的对象. 二.动机 1.在软件系统采用纯粹对象方案的问题在于大量细粒度的对象会很快充斥在系统中,从而带来很高的运行 ...

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

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

  8. 设计模式:享元(FlyWeight)模式

    设计模式:享元(FlyWeight)模式 一.前言     享元(FlyWeight)模式顾名思义,既是轻量级的,原因就是享元,共享元素,这里的元素指的是对象.如何共享对象,那就是在检测对象产生的时候 ...

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

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

最新文章

  1. ner pytorch project code
  2. Atitit.png 图片不能显示 php环境下
  3. 云服务器安装虚拟机方法,云服务器安装虚拟机方法
  4. 主板rs232接口测试软件,简洁的RS232串口通信电路与串口通信测试程序
  5. IIS处理并发请求时出现的问题及解决
  6. 机器学习:随机森林算法及其实现
  7. java/android 做题中整理的碎片小贴士(12)
  8. java8的日期API总结(JSR310)
  9. Referenced file contains errors (http://JAVA.sun.com/xml/ns/j2ee/web-app_2_5.xsd).
  10. mysql killing slave_MySQL Slave 触发 oom-killer解决方法
  11. 华为服务器装系统一直在读盘,系统重装一直在启动服务器
  12. Maven常用插件配置和使用
  13. MySQL是怎样运行的(实体书扫描+掘金小册)免费下载
  14. 计算机网络攻防技术的分析与研究
  15. SQL创建触发器以及触发器的使用实例+详解
  16. 坦克世界怎么显示服务器准心,坦克世界8.0环境设置详细教程
  17. 三菱iQ-R系列PLC控制系统项目全套资料
  18. JAVA毕业设计计算机类专业考研交流学习平台计算机源码+lw文档+系统+调试部署+数据库
  19. Keil中文显示设置
  20. Halcon找圆系列(4)测量圆直径/半径的方法之暴力拟合法 vs 测量工具法

热门文章

  1. Java虚拟机字节码指令概述
  2. SSDT表与ShadowSSDT表
  3. hdu2100(大数加)
  4. Windows Azure Storage (19) 再谈Azure Block Blob和Page Blob
  5. iOS 开发一定要尝试的 Texture(ASDK)
  6. Core Java(一)
  7. 重写与重载的区别 以及 重写中super的使用
  8. Linux下搭建mpi集群(ubuntu下用虚拟机测试)
  9. HTML td 标签的 colspan 属性
  10. Ajax异步请求-简单模版