這貨是從 Martin 大神的《公司專屬應用程式架構模式》中學到的,輔助 PHP 動態語言的特性,可以比 Java 輕鬆很多的實現消極式載入(LazyLoad)。基本原理是通過一個虛代理(Virtual Proxy)做預留位置,一旦訪問代理對象的某成員(方法或屬性),載入就被觸發。
不過我實現的這個版本有局限性:
只適用於對象,無法代理數組等基礎資料型別 (Elementary Data Type)(需要用 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();
}
}
}