懂得分享的享元模式(Flyweight Pattern)

今天学习一个对php来说意义不大的模式——享元模式

竟然意义不大为什么要学?我一直认为设计模式是不针对语言的,相对于目前的php意义不大,并不表示其本身没有意义,其在编译型的语言里还是有很多的用处的。

 

享元模式的定义

享元模式是池技术的重要实现方式,其定义为:使用共享对象可有效地支持大量的细粒度的对象。太简单了,通俗地说就是把经常要使用到的对象共享起来,而不去从新创建而占用内存。其大致构成:

1、抽象享元(Flyweight)角色

此角色是所有的具体享元类的超类,为这些类规定出需要实现的公共接口。那些需要外蕴状态的操作可以通过调用商业以参数形式传入

2、具体享元(ConcreteFlyweight)角色

实现Flyweight接口,并为内部状态(如果有的话)拉回存储空间。ConcreteFlyweight对象必须是可共享的。它所存储的状态必须是内部的

3、不共享的具体享元(UnsharedConcreteFlyweight)角色

并非所有的Flyweight子类都需要被共享。Flyweigth使共享成为可能,但它并不强制共享。

4、享元工厂(FlyweightFactory)角色

负责创建和管理享元角色。本角色必须保证享元对象可能被系统适当地共享

5、客户端(Client)角色

本角色需要维护一个对所有享元对象的引用。本角色需要自行存储所有享元对象的外部状态

 

其类图如下:


太easy了,实现代码如下:

 

<?php
/**
 *  抽象享元角色
 */
abstract class Flyweight {
	abstract public function operation( $state );
}
/**
 * 具体享元角色
 */
class ConcreteFlyweight extends Flyweight {
	private $_intrinsicState = null;
	public function __construct( $state ) {
		$this->_intrinsicState = $state;
	}
	public function operation( $state ) {
		echo 'ConcreteFlyweight operation, Intrinsic State = ' . $this->_intrinsicState. ' Extrinsic State = ' . $state . "\n";
	}
}
/**
 * 不共享的具体享元,客户端直接调用
 */
class UnsharedConcreteFlyweight extends Flyweight {
	private $_intrinsicState = null;
	public function __construct( $state ) {
		$this->_intrinsicState = $state;
	}
	public function operation( $state ) {
		echo 'UnsharedConcreteFlyweight operation, Intrinsic State = ' . $this->_intrinsicState. ' Extrinsic State = ' . $state . "\n";
	}
}
/**
 * 享元工厂角色
 */
class FlyweightFactory {
	private $_flyweights;
	public function __construct() {
		$this->_flyweights = array();
	}
	public function getFlyweigth( $state ) {
		if ( isset( $this->_flyweights[$state] ) ) {
			return $this->_flyweights[$state];
		} else {
			return $this->_flyweights[$state] = new ConcreteFlyweight( $state );
		}
	}
}
$flyweightFactory = new FlyweightFactory();
$flyweight = $flyweightFactory->getFlyweigth( 'state A' );
$flyweight->operation( 'other state A' );
$flyweight = $flyweightFactory->getFlyweigth( 'state B' );
$flyweight->operation( 'other state B' );
/* 不共享的对象,单独调用 */
$uflyweight = new UnsharedConcreteFlyweight( 'state A' );
$uflyweight->operation( 'other state A' );

?>
运行结果:
ConcreteFlyweight operation, Intrinsic State = state A Extrinsic State = other state A
ConcreteFlyweight operation, Intrinsic State = state B Extrinsic State = other state B
UnsharedConcreteFlyweight operation, Intrinsic State = state A Extrinsic State = other state A
[Finished in 0.3s]

 

 

 

享元模式的优点

享元模式可以大大减少应用程序创建的对象,降低程序内存的占用,增强程序的性能,但它同时也提高了系统的复杂性,需要分离出外部状态和内部状态,而且外部状态具有固化特性,不应该随内部状态改变而改变,否则导致系统的逻辑混乱。

 

 

享元模式的使用场景

1、系统中存在大量的相似对象

2、细粒度的对象都具备较接近的外部状态,而且内部状态与环境无关,也就是说对象没有特定身份。

3、需要缓冲池的场景

 

享元模式的扩展

1、结合单例模式

2、Composite模式

复合享元模式实际上是单纯享元模式与合成模式的组合。单纯享元对象可以作为树叶对象来讲,是可以共享的,而复合享元对象可以作为树枝对象,因此在复合享元角色中可以添加聚集管理方法

 

<?php
abstract class Flyweight {
	abstract public function operation( $state );
}
/**
 * * 具体享元角色
 */
class ConcreteFlyweight extends Flyweight {
	private $_intrinsicState = null;
	public function __construct( $state ) {
		$this->_intrinsicState = $state;
	}
	public function operation( $state ) {
		echo 'ConcreteFlyweight operation, Intrinsic State = ' . $this->_intrinsicState . ' Extrinsic State = ' . $state . "\n";
	}
}
/**
 * * 不共享的具体享元,客户端直接调用
 */
class UnsharedConcreteFlyweight extends Flyweight {
	private $_flyweights;
	public function __construct() {
		$this->_flyweights = array();
	}
	public function operation( $state ) {
		foreach ( $this->_flyweights as $flyweight ) {
			$flyweight->operation( $state );
		}
	}
	public function add( $state, Flyweight $flyweight ) {
		$this->_flyweights[$state] = $flyweight;
	}
}
/**
 * * 享元工厂角色
 */
class FlyweightFactory {
	private $_flyweights;
	public function __construct() {
		$this->_flyweights = array();
	}
	public function getFlyweigth( $state ) {
		if ( is_array( $state ) ) {
			//  复合模式
			$uFlyweight = new UnsharedConcreteFlyweight();
			foreach ( $state as $row ) {
				$uFlyweight->add( $row, $this->getFlyweigth( $row ) );
			}
			return $uFlyweight;
		} else if ( is_string( $state ) ) {
				if ( isset( $this->_flyweights[$state] ) ) {
					return $this->_flyweights[$state];
				} else {
					return $this->_flyweights[$state] = new ConcreteFlyweight( $state );
				}
			} else {
			return null;
		}
	}
}
$flyweightFactory = new FlyweightFactory();
$flyweight = $flyweightFactory->getFlyweigth( 'state A' );
$flyweight->operation( 'other state A' );

$flyweight = $flyweightFactory->getFlyweigth( 'state B' );
$flyweight->operation( 'other state B' );

/* 复合对象*/
$uflyweight = $flyweightFactory->getFlyweigth( array( 'state A', 'state B' ) );
$uflyweight->operation( 'other state A' );
?>
运行结果:
ConcreteFlyweight operation, Intrinsic State = state A Extrinsic State = other state A
ConcreteFlyweight operation, Intrinsic State = state B Extrinsic State = other state B
ConcreteFlyweight operation, Intrinsic State = state A Extrinsic State = other state A
ConcreteFlyweight operation, Intrinsic State = state B Extrinsic State = other state A
[Finished in 0.2s]

 

 

享元模式主要解决的是对象的共享问题,如何建立多个可共享的细粒度对象则是其关注的重点。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是组合模式、装饰器模式、外观模式、享元模式和代理模式的应用案例和代码实现步骤的简要说明: 1. 组合模式 (Composite Pattern): 应用案例:文件系统的目录结构可以使用组合模式来表示,其中目录和文件都可以作为容器或叶子节点,可以方便地进行递归操作。 代码实现步骤:创建一个抽象类或接口表示组件,其中包含添加、删除和获取子组件的方法。实现类分别表示叶子节点和容器节点,容器节点可以包含其他组件。 2. 装饰器模式 (Decorator Pattern): 应用案例:在一个图形绘制软件中,可以使用装饰器模式来实现不同的图形对象以及对图形进行装饰,例如添加颜色、添加边框等。 代码实现步骤:创建一个抽象类或接口表示基本对象或装饰器,其中包含一个基本对象的引用。具体装饰器类继承自该抽象类,并在调用方法时添加额外的功能。 3. 外观模式 (Facade Pattern): 应用案例:在一个电子商务平台中,可以使用外观模式来创建一个统一的接口,将不同子系统的功能封装起来,便于客户端调用。 代码实现步骤:创建一个外观类,该类提供了一个简单的接口来调用多个子系统的功能,并在内部进行协调和管理。 4. 享元模式 (Flyweight Pattern): 应用案例:在一个游戏中,可以使用享元模式来共享不同的游戏资源对象,例如共享相同的纹理、音频等,以减少内存的使用。 代码实现步骤:创建一个享元工厂类来管理共享对象,通过池化技术来缓存和重用对象,并提供一个获取共享对象的方法。 5. 代理模式 (Proxy Pattern): 应用案例:在一个网络请求中,可以使用代理模式来代表真实的网络请求对象,以进行一些额外的操作,例如鉴权、缓存等。 代码实现步骤:创建一个接口或抽象类来表示真实对象和代理对象,代理对象持有一个真实对象的引用,并在调用方法时进行一些额外的处理。 以上是这些设计模式的简要应用案例和代码实现步骤。在实际开发中,可以根据具体需求选择合适的设计模式,并根据设计模式的原则进行设计和实现。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值