享元模式,“享元”这两个字在中文里其实并没有什么特殊的意思,所以我们要把它拆分来看。“享”就是共享,“元”就是元素,这样一来似乎就很容易理解了,共享某些元素嘛。

Gof类图及解释

GoF定义:运用共享技术有效地支持大量细粒度的对象

GoF类图

代码实现

interface Flyweight
{
public function operation($extrinsicState) : void;
} class ConcreteFlyweight implements Flyweight
{
private $intrinsicState = 101;
function operation($extrinsicState) : void
{
echo '共享享元对象' . ($extrinsicState + $this->intrinsicState) . PHP_EOL;
}
} class UnsharedConcreteFlyweight implements Flyweight
{
private $allState = 1000;
public function operation($extrinsicState) : void
{
echo '非共享享元对象:' . ($extrinsicState + $this->allState) . PHP_EOL;
}
}

定义共享接口以及它的实现,注意这里有两个实现,ConcreteFlyweigh进行状态的共享,UnsharedConcreteFlyweight不共享或者说他的状态不需要去共享

class FlyweightFactory
{
private $flyweights = []; public function getFlyweight($key) : Flyweight
{
if (!array_key_exists($key, $this->flyweights)) {
$this->flyweights[$key] = new ConcreteFlyweight();
}
return $this->flyweights[$key];
}
}

保存那些需要共享的对象,做为一个工厂来创建需要的共享对象,保证相同的键值下只会有唯一的对象,节省相同对象创建的开销

$factory = new FlyweightFactory();

$extrinsicState = 100;
$flA = $factory->getFlyweight('a');
$flA->operation(--$extrinsicState); $flB = $factory->getFlyweight('b');
$flB->operation(--$extrinsicState); $flC = $factory->getFlyweight('c');
$flC->operation(--$extrinsicState); $flD = new UnsharedConcreteFlyweight();
$flD->operation(--$extrinsicState);

客户端的调用,让外部状态$extrinsicState能够在各个对象之间共享

  • 有点意思吧,这个模式的代码量可不算少
  • 当一个应用程序使用了大量非常相似的的对象,对象的大多数状都可变为外部状态时,很适合享元模式
  • 这里的工厂是存储对象列表的,不是像工厂方法或者抽象工厂一样去创建对象的,虽说这里也进行了创建,但如果对象存在,则会直接返回,而且列表也是一直维护的
  • 享元模式在现实中,大家多少一定用过,各种池技术就是它的典型应用:线程池、连接池等等,另外两个一样的字符串String类型在php或Java中都是可以===的,这也运用到了享元模式,它们连内存地址都是一样的,这不就是一种共享嘛
  • 关于享元模式,有一个极其经典的例子,比我下面的例子要好的多,那就是关于围棋的棋盘。围棋只有黑白两色,所以两个对象就够了,接下来呢?改变他们的位置状态就好啦!有兴趣的朋友可以搜搜哈!
  • Laravel中的IoC容器可以看作是一种享元模式的实现。它把对象保存在数组中,在需要的时候通过闭包机制进行取用,也有一些类有共享一些状态属性的内容。大家可以翻看代码了解了解。

还是说到科技以换壳为本这件事上。毕竟,大家都还是喜欢各种颜色的手机来彰显自己的个性。之前说过,如果每种颜色我们都要做一条生产线的话那岂不是一项巨大的投入。还好,每个型号我们的工厂(享元工厂)只生产最基本的背景壳(对象),然后通过专门的印刷线(状态变化)来进行上色不就好啦!嗯,下一款Iphone早晚也会模仿我们的,看来我们得先把各种金、各种土豪色集齐才行,说不定还能召唤神龙呢!!

完整代码:https://github.com/zhangyue0503/designpatterns-php/blob/master/13.flyweights/source/flyweights.php

实例

果然不出意外的我们还是来发短信,这回的短信依然使用的阿里云和极光短信来进行发送,不过这次我们使用享元模式来实现,这里的享元工厂我们保存了两种不同类型的对象哦,通过内外状态来让它们千变万化吧!

短信发送类图

完整源码:https://github.com/zhangyue0503/designpatterns-php/blob/master/13.flyweights/source/flyweights-message.php


<?php interface Message
{
public function send(User $user);
} class AliYunMessage implements Message
{
private $template;
public function __construct($template)
{
$this->template = $template;
}
public function send(User $user)
{
echo '使用阿里云短信向' . $user->GetName() . '发送:';
echo $this->template->GetTemplate(), PHP_EOL;
}
} class JiGuangMessage implements Message
{
private $template;
public function __construct($template)
{
$this->template = $template;
}
public function send(User $user)
{
echo '使用极光短信向' . $user->GetName() . '发送:';
echo $this->template->GetTemplate(), PHP_EOL;
}
} class MessageFactory
{
private $messages = [];
public function GetMessage(Template $template, $type = 'ali')
{
$key = md5($template->GetTemplate() . $type);
if (!key_exists($key, $this->messages)) {
if ($type == 'ali') {
$this->messages[$key] = new AliYunMessage($template);
} else {
$this->messages[$key] = new JiGuangMessage($template);
}
}
return $this->messages[$key];
} public function GetMessageCount()
{
echo count($this->messages);
}
} class User
{
public $name;
public function GetName()
{
return $this->name;
}
} class Template
{
public $template;
public function GetTemplate()
{
return $this->template;
}
} // 内部状态
$t1 = new Template();
$t1->template = '模板1,不错哟!'; $t2 = new Template();
$t2->template = '模板2,还好啦!'; // 外部状态
$u1 = new User();
$u1->name = '张三'; $u2 = new User();
$u2->name = '李四'; $u3 = new User();
$u3->name = '王五'; $u4 = new User();
$u4->name = '赵六'; $u5 = new User();
$u5->name = '田七'; // 享元工厂
$factory = new MessageFactory(); // 阿里云发送
$m1 = $factory->GetMessage($t1);
$m1->send($u1); $m2 = $factory->GetMessage($t1);
$m2->send($u2); echo $factory->GetMessageCount(), PHP_EOL; // 1 $m3 = $factory->GetMessage($t2);
$m3->send($u2); $m4 = $factory->GetMessage($t2);
$m4->send($u3); echo $factory->GetMessageCount(), PHP_EOL; // 2 $m5 = $factory->GetMessage($t1);
$m5->send($u4); $m6 = $factory->GetMessage($t2);
$m6->send($u5); echo $factory->GetMessageCount(), PHP_EOL; // 2 // 加入极光
$m1 = $factory->GetMessage($t1, 'jg');
$m1->send($u1); $m2 = $factory->GetMessage($t1);
$m2->send($u2); echo $factory->GetMessageCount(), PHP_EOL; // 3 $m3 = $factory->GetMessage($t2);
$m3->send($u2); $m4 = $factory->GetMessage($t2, 'jg');
$m4->send($u3); echo $factory->GetMessageCount(), PHP_EOL; // 4 $m5 = $factory->GetMessage($t1, 'jg');
$m5->send($u4); $m6 = $factory->GetMessage($t2, 'jg');
$m6->send($u5); echo $factory->GetMessageCount(), PHP_EOL; // 4

说明

  • 代码有点多吧,但其实一共是两种类型的类,生成了四种对象。这里每个类不同的对象是根据模板来区分的
  • 这样的组合还是比较方便的吧,再结合别的模式将工厂这里优化一下,嗯,前途不可限量,你们可以想想哦!
  • 享元模式适用于系统中存在大量的相似对象以及需要缓冲池的场景,能够降低内存占用,提高效率,但会增加复杂度,需要分享内外部状态
  • 最主要的特点是有一个唯一标识,当内存中已经有这个对象了,直接返回对象,不用再去创建了

下期看点

享元模式的例子说得有点牵强,不过确实这类设计模式在我们日常的开发中一方面用得不多,另一方面典型的例子又太经典,反正只要记住它的特点就好了,具体在应用的时候说不准写完代码回头一看你会发现这不就是我学过的享元模式嘛!好了,下一篇我们学习一个也是比较少用而且复杂的模式,但是也许你也能经常见到哦!组合模式

关注公众号:【硬核项目经理】获取最新文章

添加微信/QQ好友:【xiaoyuezigonggong/149844827】免费得PHP、项目管理学习资料

知乎、公众号、抖音、头条搜索【硬核项目经理】

B站ID:482780532

PHP设计模式之享元模式的更多相关文章

  1. 乐在其中设计模式(C#) - 享元模式(Flyweight Pattern)

    原文:乐在其中设计模式(C#) - 享元模式(Flyweight Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 享元模式(Flyweight Pattern) 作者:weba ...

  2. python设计模式之享元模式

    python设计模式之享元模式 由于对象创建的开销,面向对象的系统可能会面临性能问题.性能问题通常在资源受限的嵌入式系统中出现,比如智能手机和平板电脑.大型复杂系统中也可能会出现同样的问题,因为要在其 ...

  3. 【GOF23设计模式】享元模式

    来源:http://www.bjsxt.com/ 一.[GOF23设计模式]_享元模式.享元池.内部状态.外部状态.线程池.连接池 package com.test.flyweight; /** * ...

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

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

  5. Head First设计模式之享元模式(蝇量模式)

    一.定义 享元模式(Flyweight Pattern)主要用于减少创建对象的数量,以减少内存占用和提高性能.这种类型的设计模式属于结构型模式,它提供了减少对象数量从而改善应用所需的对象结构的方式. ...

  6. 【Unity3D与23种设计模式】享元模式(Flyweight)

    GoF中定义: "使用共享的方式,让一大群小规模对象能更有效地运行" 享元模式一般应用在游戏角色属性设置上 游戏策划需要通过"公式计算"或者"实际测试 ...

  7. Java进阶篇设计模式之七 ----- 享元模式和代理模式

    前言 在上一篇中我们学习了结构型模式的组合模式和过滤器模式.本篇则来学习下结构型模式最后的两个模式, 享元模式和代理模式. 享元模式 简介 享元模式主要用于减少创建对象的数量,以减少内存占用和提高性能 ...

  8. Java设计模式之七 ----- 享元模式和代理模式

    前言 在上一篇中我们学习了结构型模式的组合模式和过滤器模式.本篇则来学习下结构型模式最后的两个模式, 享元模式和代理模式. 享元模式 简介 享元模式主要用于减少创建对象的数量,以减少内存占用和提高性能 ...

  9. 【设计模式】享元模式(Flyweight)

    摘要: 1.本文将详细介绍享元模式的原理和实际代码中特别是Android系统代码中的应用. 纲要: 1. 引入享元模式 2. 享元模式的概念及优缺点介绍 3. 享元模式在Android源码中的应用 1 ...

  10. 设计模式之享元模式(PHP实现)

    github地址:https://github.com/ZQCard/design_pattern /** * 减少创建对象的数量,以减少内存占用和提高性能.这种类型的设计模式属于结构型模式, * 它 ...

随机推荐

  1. 客户端连接mysql数据库反应慢

    远程客户端连接MysqL数据库太慢解决方案 局域网客户端访问mysql 连接慢问题解决 编辑mysql配置文件 # vi my.conf [mysqld] skip-name-resolve 重启my ...

  2. 常见web中间件漏洞(五)weblogic漏洞

    继续整理有关中间件漏洞思路(仅做简单思路整理,不是复现,复现请参考大佬们的长篇好文,会在文章中列举部分操作) WebLogic是Oracle公司出品的一个application server,确切的说 ...

  3. Linux提权手法整理

    之前写过了windows提权小结,这下一篇水什么就有了嘛,于是有了这篇水文,整理一下Linux提权 前篇windows提权小结 ,链接送上 https://www.cnblogs.com/lcxblo ...

  4. SQL 练习38

    查询选修了全部课程的学生信息 SELECT * from Student WHERE SId IN ( SELECT sid from sc GROUP BY sid HAVING COUNT(cid ...

  5. Mysql生成UUID的SQL语句

    大写的UUID: SELECT UPPER(UUID()); 小写的UUID: SELECT LOWER(UUID()); SELECT UUID(); 去掉横杠的UUID: SELECT REPLA ...

  6. windows和liunx下换行符问题

    区别 windows换行符是: \r\n liunx换行符是: \n 问题 程序处理的时候就会有问题,因为在Windows的文件多了一个\r 解决办法(转换文件格式) vim file :set fi ...

  7. 配置之XML--读取XML文件 转存为Key-Value

    将XML文件读取 绑定数据至Dictionary Eg: Xml文件 <?xml version="1.0" encoding="utf-8" ?> ...

  8. 由struts2中配置使用servlet引发的思考和复习

    Struts2拦截器到底拦截了什么? 关于struts2中的拦截器,首先再次理解其实只能过滤其中访问的action的映射!再者,因为struts中的action其实就是起到替代servlet作用的,所 ...

  9. mzy,struts学习(一)

    大家都在讲struts已经过时了,现在都是前后台分离,没有必要去学一个淘汰的框架,但是怎么讲呢?我觉得,struts能够流行那么多年,肯定有它的原因,肯定有很多优秀和好的地方,有一个指导过我的人给我讲 ...

  10. Vivado实战—单周期CPU指令分析

    引言   不知道你是否和我有过同样的感受,<计算机组成原理>这门学科学起来如此的艰难:一节课下来,教室黑板上留下了满满的 "足迹",看上去也挺简单的,不就是 0 和 1 ...