以下文字部分均为本人自己的想法与总结所以未必正确;
代码摘自https://blog.csdn.net/rust94/article/details/88895465
享元模式:
概念: 运用共享技术有效地支持大量细粒度对象的复用。系统只使用少量的对象,而这些对象都很相似,状态变化很小,可以实现对象的多次复用。由于享元模式要求能够共享的对象必须是细粒度对象,因此它又称为轻量级模式,它是一种对象结构型模式。
角色:
- Flyweight: 抽象享元类
- ConcreteFlyweight: 具体享元类
- UnsharedConcreteFlyweight: 非共享具体享元类
- FlyweightFactory: 享元工厂类
个人理解:
享元模式的享元工厂类能保证具体享元对象在内存中只存有一份,避免创建大量相同的对象而消耗内存。
享元模式与单例模式的区别:
- 单例模式是类级别的,一个类只能有一个对象实例;
- 享元模式是对象级别的,可以有多个对象实例,多个变量引用同一个对象实例;
- 享元模式主要是为了节约内存空间,提高系统性能,而单例模式主要为了可以共享数据;
适用场景:
- 系统中存在大量相似对象
- 需要缓冲池的场景,例如数据库连接池
网上找来的一张结构图:
<?php
class User
{
private $name;
function __construct($name)
{
$this->name = $name;
}
public function getName()
{
return $this->name;
}
}
//享元抽象类
abstract class WebSite
{
abstract public function use(User $user);
}
// 具体网站类(具体享元实现类)
class ConcreteWebSite extends WebSite
{
private $name = '';
function __construct($name)
{
$this->name = $name;
}
public function use(User $user)
{
echo "网站分类: ".$this->name."用户:".$user->getName()."\n";
}
}
//不共享的具体享元类,客户端直接调用
class UnsharedConcreteWebSite extends WebSite {
private $name = '';
function __construct($name)
{
$this->name = $name;
}
public function use(User $user)
{
echo "网站分类: ".$this->name."用户:".$user->getName()."\n";
}
}
//网站工厂(享元工厂类)
class WebSiteFactory
{
private $flyweights = [];
public function getWebSiteGategory($key)
{
if (empty($this->flyweights[$key])) {
$this->flyweights[$key] = new ConcreteWebSite($key);
}
return $this->flyweights[$key];
}
public function getWebSiteCount()
{
return count($this->flyweights);
}
}
$f = new WebSiteFactory();
$fx = $f->getWebSiteGategory('产品展示');
$fx->use(new User('张伟'));
$fy = $f->getWebSiteGategory('产品展示');
$fy->use(new User('王伟'));
$fz = $f->getWebSiteGategory('产品展示');
$fz->use(new User('王芳'));
$fl = $f->getWebSiteGategory('博客');
$fl->use(new User('李伟'));
$fm = $f->getWebSiteGategory('博客');
$fm->use(new User('王秀英'));
$fn = $f->getWebSiteGategory('博客');
$fn->use(new User('李秀英'));
//非共享享元对象
$fo = new UnsharedConcreteWebSite('博客');
$fo->use(new User("小明"));
//查看享元对象总数
echo "网站分类总数:".$f->getWebSiteCount();