概述
在我們開始實現Dynamic Proxy之前, 也許我們應該先了解一下什麼是Proxy和它有什麼用. 下面是一篇來自博客堂加菲貓的不錯的形象講述Proxy的文章: “武當學藝與緩存代理”. Proxy模式是”GoF”介紹的23個設計模式之一, Proxy的目的是” Provide a surrogate or placeholder for another object to control access(小型網站之最愛) to it(為其他對象提供一種代理以控制對這個對象的訪問)”. 一般常見的代理模式有: 遠程代理(Remote Proxy), 虛代理(Virtual Proxy), 保護代理(Protection Proxy), 智能代理(Smart Proxy).
但是使用代理有一個壞處就是你得手動創建所需要代理類的副本(即代理類). 這意味著如果你為Image類創建一個Virtual Proxy, 那麼你不得不手動創建一個與Image類有相同Method的ImageProxy類. Ok, 如果你和我一樣懶, 你一定會想到動態來產生Proxy. 是的, 接下來你就會發現, 其實在php(做為現在的主流開發語言)4中你很容易實現它.
實現
由於php(做為現在的主流開發語言)4是解釋型語言, 弱類型, 且無接口. 所以在實現的時候既有方便之處又有不妥之處. 此處不拘泥於實現方法, 本文也只是實現方法之一.
本文實現的策略其實非常簡單. 核心即ProxyFactory類及Clazz類, ProxyFactory負責實例化Clazz, 並賦值. 而由Clazz類負責創建並返回Proxy. 創建Proxy是以寫入臨時文件方式進行的.
具體請查看ProxyFactory.php(做為現在的主流開發語言)和Clazz.php(做為現在的主流開發語言)二個文件中的代碼. 此處不再贅述.
另外在ProxyInvocationHandler.php(做為現在的主流開發語言)中我們定義了一個ProxyInvocationHandler類.
示例
我們現在有一個ReadFileClass類, 該類繼承自IReadFileClass, 由於php(做為現在的主流開發語言)4沒有接口, 所以此處接口算是模擬的, 事實上在php(做為現在的主流開發語言)4中不使用實現接口也是可行的J. 二個類的具體內容請見清單一和清單二.
清單一
class IReadFileClass
{
function ReadMyFile() {}
}
清單二
class ReadFileClass extends IReadFileClass
{
function ReadMyFile()
{
$fp = fopen(test.txt, "r");
$data = fread($fp, filesize(test.txt));
fclose($fp);
return $data;
}
}
OK, 我們現在要加入驗證用戶的功能, 即為ReadFileClass中的方法加入保護控制. 如果采用手動創建代理, 那麼你可以繼承ReadFileClass或者實現IReadFileClass, 並加入保護代碼(其實在php(做為現在的主流開發語言)4中甚為自由, 因為除了基本類型外都是object-_-). 不過我們現在試試用剛才實現的動態代理來創建Proxy.
請看清單三的ReadFileClassProxy的代碼, 注意該類繼承自ProxyInvocationHandler類.
清單三
require_once(ProxyFactory.php(做為現在的主流開發語言));
require_once(ProxyInvocationHandler.php(做為現在的主流開發語言));
require_once(Auth.php(做為現在的主流開發語言));
class ReadFileClassProxy extends ProxyInvocationHandler
{
var $object;
function ReadFileClassProxy(&$obj)
{
$this->object = &$obj;
}
//
function NewInstance(&$obj)
{
$proxyFactory = ProxyFactoryInstance();
return $proxyFactory->create(new ReadFileClassProxy(&$obj),
get_parent_class(&$obj));
}
// $proxy is not used here, but it is useful.
function Invoke(&$proxy, $method, $parameters)
{
$uname = Binzy;
//$uname = Jasmin;
if (Auth::CheckAuth($uname))
{
Return parent::Invoke(&$proxy, $method, $parameters);
}
else
{
//
return No Permission!;
}
}
}
Auth類是一個進行權限驗證的類, 此處我們只是簡單的查看傳入的UserName, 如果是Binzy, 那麼自然是可以看秘密的J, 如果是Jasmin, 那麼HoHo, 沒得看, 給Binzy點空間嘛.:D 詳見清單四.