這貨是從 Martin 大神的《企業應用架構模式》中學到的,輔助 PHP 動態語言的特性,可以比 Java 輕松很多的實現延遲加載(LazyLoad)。基本原理是通過一個虛代理(Virtual Proxy)做占位符,一旦訪問代理對象的某成員(方法或屬性),加載就被觸發。
不過我實現的這個版本有局限性:
只適用於對象,無法代理數組等基本數據類型(需要用 ArrayObject 一類的內置對象封裝)
被代理之後,一些帶有操作符重載性質的接口實現就失效了,例如 ArrayAccess 的索引器、Itreator 的迭代器,如果是用該代理來處理集合類型的延遲加載,還需要繼承一個子類做特殊處理,以便外部用 foreach 迭代
demo
復制代碼 代碼如下:
// 測試
$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);
Virtual Proxy
復制代碼 代碼如下:
/**
* 虛代理,只有在被訪問成員時才調用閉包函數生成目標對象。
*
* @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();
}
}
}