PHP設(shè)計(jì)模式漫談之代理模式
原創(chuàng)【51CTO獨(dú)家特稿】設(shè)計(jì)模式( Design Pattern)是從建筑設(shè)計(jì)領(lǐng)域引入到計(jì)算機(jī)科學(xué)的。設(shè)計(jì)模式是對(duì)軟件設(shè)計(jì)中普遍存在(且反復(fù)出現(xiàn))的各種問(wèn)題,所提出的解決方案。設(shè)計(jì)模式并不直接用來(lái)完成程序碼的編寫,而是描述在各種不同情況下,要怎么解決問(wèn)題的一種方案。
更多關(guān)于PHP設(shè)計(jì)模式方面的入門與應(yīng)用可以參考51CTO之前的報(bào)道《使用設(shè)計(jì)模式改善程序結(jié)構(gòu)》以及《架構(gòu)、框架、設(shè)計(jì)模式之間的關(guān)系簡(jiǎn)述》。
我們經(jīng)??吹疥P(guān)于Java和.NET平臺(tái)上設(shè)計(jì)模式的論述和講解,其實(shí),在PHP 5對(duì)面向?qū)ο蟮闹С指油晟浦?,設(shè)計(jì)模式的應(yīng)用也可引入到PHP中并發(fā)揮重要作用。51CTO.com將從本周起以每周一期的形式連載《PHP設(shè)計(jì)模式漫談》的系列文章,以理論與代碼實(shí)例相結(jié)合的方式講解PHP中的設(shè)計(jì)模式。希望對(duì)從事PHP研發(fā)的讀者有所幫助。
今天我們要談的是PHP設(shè)計(jì)模式中的代理模式(Proxy),它是對(duì)簡(jiǎn)單處理程序(或指針)的增強(qiáng),用于引用一個(gè)對(duì)象:這個(gè)指針被代理(Proxy)對(duì)象取代,代理對(duì)象位于客戶端(Client)和真實(shí)執(zhí)行程序之間,指針有一個(gè)可被多個(gè)目標(biāo)利用的鉤子。
從技術(shù)上講,這種模式在客戶端和真實(shí)主體(RealSubject)之間插入一個(gè)代理對(duì)象,維護(hù)subject接口和用不同的方式委派它的方法。代理可以透明地做任何事情:懶散創(chuàng)建RealSubject或載入數(shù)據(jù),與其它機(jī)器交換消息,寫時(shí)復(fù)制策略等。這與HTTP代理有點(diǎn)類似,其客戶端(如瀏覽器)和應(yīng)用程序依賴于與HTTP服務(wù)器的聯(lián)系,代理在管理連接時(shí)可以完成其它任務(wù),如訪問(wèn)控制和緩存大型下載文件。
代理模式的對(duì)象圖與裝飾模式對(duì)象圖在結(jié)構(gòu)上類似,但表達(dá)的目的各有不同,裝飾者給對(duì)象動(dòng)態(tài)增加行為,而代理則控制來(lái)自客戶端的訪問(wèn)。此外,代理只在需要時(shí)才創(chuàng)建RealSubject。
參與者:
◆客戶端(Client):取決于主體(Subject)實(shí)現(xiàn);
◆主體(Subject):RealSubject的抽象;
◆真實(shí)主體(RealSubject):完成代價(jià)高昂的工作或包含大量的數(shù)據(jù);
◆代理(Proxy):為Client提供一個(gè)與Subject一致的引用,僅在需要時(shí)才創(chuàng)建RealSubject實(shí)例或與RealSubject實(shí)例通信。
下面是兩個(gè)被廣泛使用的代理模式例子:
1、對(duì)象-關(guān)系映射(Orms)在運(yùn)行中創(chuàng)建代理作為實(shí)體類的子類,以實(shí)現(xiàn)懶散加載(虛擬代理),這個(gè)代理會(huì)覆蓋所有實(shí)體方法,在前面追加一個(gè)載入程序,在方法被真正調(diào)用前不會(huì)包含任何數(shù)據(jù),Orms代理支持對(duì)象間的雙向關(guān)系,不用加載整個(gè)數(shù)據(jù)庫(kù),因?yàn)樗鼈儽恢糜诋?dāng)前加載對(duì)象圖的邊界。
2、Java RMI使用遠(yuǎn)程代理對(duì)象(遠(yuǎn)程代理),當(dāng)它們的方法被調(diào)用時(shí),代理序列化參數(shù),執(zhí)行網(wǎng)絡(luò)上的請(qǐng)求,委托調(diào)用另一個(gè)節(jié)點(diǎn)上的真實(shí)對(duì)象,這種技術(shù)允許透明地調(diào)用遠(yuǎn)程對(duì)象,不用擔(dān)心它們是否在同一臺(tái)機(jī)器上,但這種透明度很容易會(huì)使執(zhí)行速度變慢。
下面的代碼示例實(shí)現(xiàn)了一個(gè)ImageProxy,推遲了圖像數(shù)據(jù)的加載。
- /**
- * Subject interface.
- * Client depends only on this abstraction.
- */
- interface Image
- {
- public function getWidth();
- public function getHeight();
- public function getPath();
- /**
- * @return string the image's byte stream
- */
- public function dump();
- }
- /**
- * Abstract class to avoid repetition of boilerplate code in the Proxy
- * and in the Subject. Only the methods which can be provided without
- * instancing the RealSubject are present here.
- */
- abstract class AbstractImage implements Image
- {
- protected $_width;
- protected $_height;
- protected $_path;
- protected $_data;
- public function getWidth()
- {
- return $this->_width;
- }
- public function getHeight()
- {
- return $this->_height;
- }
- public function getPath()
- {
- return $this->_path;
- }
- }
- /**
- * The RealSubject. Always loads the image, even if no dump of the data
- * is required.
- */
- class RawImage extends AbstractImage
- {
- public function __construct($path)
- {
- $this->_path = $path;
- list ($this->_width, $this->_height) = getimagesize($path);
- $this->_data = file_get_contents($path);
- }
- public function dump()
- {
- return $this->_data;
- }
- }
- /**
- * Proxy. Defers loading the image data until it becomes really mandatory.
- * This class does its best to postpone the very expensive operations
- * such as the actual loading of the BLOB.
- */
- class ImageProxy extends AbstractImage
- {
- public function __construct($path)
- {
- $this->_path = $path;
- list ($this->_width, $this->_height) = getimagesize($path);
- }
- /**
- * Creates a RawImage and exploits its functionalities.
- */
- protected function _lazyLoad()
- {
- if ($this->_realImage === null) {
- $this->_realImage = new RawImage($this->_path);
- }
- }
- public function dump()
- {
- $this->_lazyLoad();
- return $this->_realImage->dump();
- }
- }
- /**
- * Client class that does not use the data dump of the image.
- * Passing blindly a Proxy to this class and to other Clients makes sense
- * as the data would be loaded anyway when Image::dump() is called.
- */
- class Client
- {
- public function tag(Image $img)
- {
- return '
$img->getPath() . '" alt="" width="'
- . $img->getWidth() . '" height="'
- . $img->getHeight() . '" />';
- }
- }
- $path = '/home/giorgio/shared/Immagini/kiki.png';
- $client = new Client();
- $image = new RawImage($path); // loading of the BLOB takes place
- echo $client->tag($image), "\n";
- $proxy = new ImageProxy($path);
- echo $client->tag($proxy), "\n"; // loading does not take place even here
以上代碼實(shí)現(xiàn)了PHP的代理模式。簡(jiǎn)單來(lái)講,代理模式就是為其他對(duì)象提供一個(gè)代理以控制對(duì)這個(gè)對(duì)象的訪問(wèn)。
關(guān)于設(shè)計(jì)模式的其他方面
我們知道了什么是設(shè)計(jì)模式以及PHP設(shè)計(jì)模式中的代理模式。這里還有一些關(guān)于設(shè)計(jì)模式的基本概念應(yīng)該被熟知。
◆算法不能算是一種設(shè)計(jì)模式,因?yàn)樗惴ㄖ饕怯脕?lái)解決計(jì)算上的問(wèn)題,而非設(shè)計(jì)上的問(wèn)題。
◆面向?qū)ο笤O(shè)計(jì)模式通常以類別或物件來(lái)描述其中的關(guān)系和相互作用,但不涉及用來(lái)完成應(yīng)用程序的特定類別或物件。
◆設(shè)計(jì)模式主要是使不穩(wěn)定的依賴于相對(duì)穩(wěn)定、具體依賴于相對(duì)抽象,避免會(huì)引起麻煩的緊耦合,以增強(qiáng)軟件設(shè)計(jì)面對(duì)并適應(yīng)變化的能力。
◆并非所有的軟件模式都是設(shè)計(jì)模式,設(shè)計(jì)模式特指軟件設(shè)計(jì)層次上的問(wèn)題。還有其它非設(shè)計(jì)模式的模式,如架構(gòu)模式。
我們會(huì)在下周繼續(xù)為您連載《PHP設(shè)計(jì)模式漫談》,敬請(qǐng)期待。
【編輯推薦】
- 鐘勝輝談PHP發(fā)展的現(xiàn)狀和前景
- Java設(shè)計(jì)模式與應(yīng)用淺談
- 架構(gòu)、框架、設(shè)計(jì)模式之間的關(guān)系簡(jiǎn)述
- 專題:PHP開(kāi)發(fā)基礎(chǔ)入門
原文:Practical Php Patterns: Proxy
鏈接:http://giorgiosironi.blogspot.com/2010/02/practical-php-patterns-proxy.html