話說這貨是從 Martin 大神的《企業應用架構模式》中學到的,輔助 PHP 動態語言的特性,可以比 Java 輕松很多的實現延遲加載——通過一個虛代理占位符。唯一的缺陷,是只能代理對象,不能代理內置基本類型。
我試水的 PHP 領域模型設計中,也是用這個來實現 DomainObject 的延遲加載。
復制代碼 代碼如下:
* 虛代理,只有在被訪問成員時才調用閉包函數生成目標對象。
*
* @author tonyseek
*
*/
class VirtualProxy
{
private $holder = null;
private $loader = null;
/**
* 虛代理,只有在被訪問成員時才調用閉包函數生成目標對象。
*
* @param Closure $loader 生成被代理對象的閉包函數
*/
public function __construct(Closure $loader)
{
$this->loader = $loader;
}
/**
* 代理成員方法的調用
*
* @param string $method
* @param array $arguments
* @throws BadMethodCallException
* @return mixed
*/
public function __call($method, array $arguments = null)
{
$this->check();
if (!method_exists($this->holder, $method)) {
throw new BadMethodCallException();
}
return call_user_func_array(
array(&$this->holder, $method),
$arguments);
}
/**
* 代理成員屬性的讀取
*
* @param string $property
* @throws ErrorException
* @return mixed
*/
public function __get($property)
{
$this->check();
if (!isset($this->holder->$property)) {
throw new ErrorException();
}
return $this->holder->$property;
}
/**
* 代理成員屬性的賦值
*
* @param string $property
* @param mixed $value
*/
public function __set($property, $value)
{
$this->check();
$this->holder->$property = $value;
}
/**
* 檢查是否已經存在被代理對象,不存在則生成。
*/
private function check()
{
if (null == $this->holder) {
$loader = $this->loader;
$this->holder = $loader();
}
}
}
// 測試
$v = new VirtualProxy(function(){
echo 'Now, Loading', "\n";
$a = new ArrayObject(range(1,100));
$a->abc = 'a';
// 實際使用中,這裡調用的是 DataMapper 的 findXXX 方法
// 返回的是領域對象集合
return $a;
});
// 代理對象直接當作原對象訪問
// 而此時構造方法傳入的 callback 函數才被調用
// 從而實現加載對象操作的延遲
echo $v->abc . $v->offsetGet(50);