一、 摘要
本文簡單闡述了Proxy模式及具體說明了如何在PHP4中實現(xiàn)動態(tài)代理, 本文只是給出了一個實現(xiàn)的方法的原型. 由于水平有限, 有任何意見和建議請反饋給Binzy [ Binzy at JustDN dot Com ].
二、 概述
在我們開始實現(xiàn)Dynamic Proxy之前, 也許我們應(yīng)該先了解一下什么是Proxy和它有什么用. 下面是一篇來自博客堂加菲貓的不錯的形象講述Proxy的文章: “武當學藝與緩存代理”. Proxy模式是”GoF”介紹的23個設(shè)計模式之一, Proxy的目的是” Provide a surrogate or placeholder for another object to control access to it(為其他對象提供一種代理以控制對這個對象的訪問)”. 一般常見的代理模式有: 遠程代理(Remote Proxy), 虛代理(Virtual Proxy), 保護代理(Protection Proxy), 智能代理(Smart Proxy).
但是使用代理有一個壞處就是你得手動創(chuàng)建所需要代理類的副本(即代理類). 這意味著如果你為Image類創(chuàng)建一個Virtual Proxy, 那么你不得不手動創(chuàng)建一個與Image類有相同Method的ImageProxy類. Ok, 如果你和我一樣懶, 你一定會想到動態(tài)來產(chǎn)生Proxy. 是的, 接下來你就會發(fā)現(xiàn), 其實在PHP4中你很容易實現(xiàn)它.
三、 實現(xiàn)
由于PHP4是解釋型語言, 弱類型, 且無接口. 所以在實現(xiàn)的時候既有方便之處又有不妥之處. 此處不拘泥于實現(xiàn)方法, 本文也只是實現(xiàn)方法之一.
本文實現(xiàn)的策略其實非常簡單. 核心即ProxyFactory類及Clazz類, ProxyFactory負責實例化Clazz, 并賦值. 而由Clazz類負責創(chuàng)建并返回Proxy. 創(chuàng)建Proxy是以寫入臨時文件方式進行的.
具體請查看ProxyFactory.php和Clazz.php二個文件中的代碼. 此處不再贅述.
另外在ProxyInvocationHandler.php中我們定義了一個ProxyInvocationHandler類.
四、 示例
我們現(xiàn)在有一個ReadFileClass類, 該類繼承自IReadFileClass, 由于PHP4沒有接口, 所以此處接口算是模擬的, 事實上在PHP4中不使用實現(xiàn)接口也是可行的J. 二個類的具體內(nèi)容請見清單一和清單二.
清單一
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, 我們現(xiàn)在要加入驗證用戶的功能, 即為ReadFileClass中的方法加入保護控制. 如果采用手動創(chuàng)建代理, 那么你可以繼承ReadFileClass或者實現(xiàn)IReadFileClass, 并加入保護代碼(其實在PHP4中甚為自由, 因為除了基本類型外都是object-_-). 不過我們現(xiàn)在試試用剛才實現(xiàn)的動態(tài)代理來創(chuàng)建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類是一個進行權(quán)限驗證的類, 此處我們只是簡單的查看傳入的UserName, 如果是Binzy, 那么自然是可以看秘密的J, 如果是Jasmin, 那么HoHo, 沒得看, 給Binzy點空間嘛.:D 詳見清單四.
清單四
class Auth
{
function Auth()
{
}
// bool
//
function CheckAuth($username)
{
if ($username == 'Binzy')
{
return true;
}
return false;
}
}
Ok, 下面我們來使用我們創(chuàng)建的代理. 請見清單五.
清單五
require_once('ReadFileClass.php');
require_once('ReadFileClassProxy.php');
$proxy = ReadFileClassProxy::NewInstance(new ReadFileClass());
print $proxy->ReadMyFile();
結(jié)果如下:
如果是Binzy, 那么自然可以知道那個秘密.
500) this.width=500" vspace=10 border=0>
如果是Jasmin, 這個秘密當然不能讓她知道.
500) this.width=500" vspace=10 border=0>
五、 總結(jié)
代理是一個非常有用的模式. PHP4雖然并不是真正的Object-Oriented, 但仍然可以實現(xiàn)你想實現(xiàn)的設(shè)計. 寫本文的目的有很大部分是希望國內(nèi)PHP開發(fā)者不要再拘泥于現(xiàn)在的開發(fā)現(xiàn)狀, 開發(fā)出更好的PHP軟件. 而不是一堆Script的堆積.
六、 感謝
感謝好友 Freeman 為我做測試.
感謝 mmkk 的Code Formatter HTC.
七、 參考
1. GoF
2. GoF中譯本
3. PHP參考手冊 http://www.php.net/manual/en/
八、 相關(guān)下載
相關(guān)附件:本文原代碼