PHP設(shè)計(jì)模式之工廠模式(Factory)入門與應(yīng)用詳解
本文實(shí)例講述了PHP設(shè)計(jì)模式之工廠模式(Factory)。分享給大家供大家參考,具體如下:
工廠模式的意思其實(shí)就是提供獲取某個(gè)對(duì)象實(shí)例的一個(gè)接口,同時(shí)使調(diào)用代碼避免確定實(shí)例化基類的步驟,實(shí)際上就是建立一個(gè)統(tǒng)一的類實(shí)例化的函數(shù)接口,完事統(tǒng)一調(diào)用,統(tǒng)一控制,它是PHP中常用的一種設(shè)計(jì)模式,一般會(huì)配合單例模式一起使用,來(lái)加載php類庫(kù)中的類。來(lái)看一個(gè)簡(jiǎn)單的應(yīng)用場(chǎng)景:
- 我們擁有一個(gè)Json類,String類,Xml類。
- 如果我們不使用工廠方式實(shí)例化這些類,則需要每一個(gè)類都需要new一遍,過(guò)程不可控,類多了,到處都是new的身影
- 引進(jìn)工廠模式,通過(guò)工廠統(tǒng)一創(chuàng)建對(duì)象實(shí)例。
代碼如下:
<?php //工廠模式 提供獲取某個(gè)對(duì)象實(shí)例的一個(gè)接口,同時(shí)使調(diào)用代碼避免確定實(shí)例化基類的步驟 //字符串類 class String { public function write() {} } //Json類 class Json { public function getJsonData() {} } //xml類 class Xml { public function buildXml() {} } //工廠類 class Factory { public static function create($class) { return new $class; } } Factory::create("Json"); //獲取Json對(duì)象
我們現(xiàn)在應(yīng)該對(duì)于工廠模式有了一個(gè)大概的理解了,咱們接下來(lái)可以從字面上來(lái)理解一下。
工廠么,它就是生產(chǎn)產(chǎn)品的地方,它有原料,設(shè)備和產(chǎn)品,那么在PHP中,我們可以理解為,這個(gè)工廠模式可以通過(guò)一個(gè)工廠類(設(shè)備),來(lái)調(diào)用自身的靜態(tài)方法(生產(chǎn)方式)來(lái)產(chǎn)生對(duì)象實(shí)例(產(chǎn)品),在上述實(shí)例中的Json類等,就相當(dāng)于原料了。
理解了上面的一段話之后,我們就可以再深入的了解下這個(gè)工廠模式了。
我們來(lái)考慮以下場(chǎng)景,如果項(xiàng)目中,我們通過(guò)一個(gè)類創(chuàng)建對(duì)象,在快完成或者已經(jīng)完成,要擴(kuò)展功能的時(shí)候,發(fā)現(xiàn)原來(lái)的類的類名不是很合適或者發(fā)現(xiàn)類需要添加構(gòu)造函數(shù)參數(shù)才能實(shí)現(xiàn)功能擴(kuò)展,在這種情況下,大家就可以感受到“高內(nèi)聚低耦合”的博大精深,我們可以嘗試使用工廠模式解決這個(gè)問(wèn)題。
還有就是最經(jīng)典的數(shù)據(jù)庫(kù)連接問(wèn)題等等,都可以使用工廠模式來(lái)接覺(jué)問(wèn)題,咱們也不廢話,來(lái)看一下網(wǎng)上一個(gè)比較經(jīng)典的案例:
interface Transport{ public function go(); } class Bus implements Transport{ public function go(){ echo "bus每一站都要停"; } } class Car implements Transport{ public function go(){ echo "car跑的飛快"; } } class Bike implements Transport{ public function go(){ echo "bike比較慢"; } } class transFactory{ public static function factory($transport) { switch ($transport) { case 'bus': return new Bus(); break; case 'car': return new Car(); break; case 'bike': return new Bike(); break; } } } $transport=transFactory::factory('car'); $transport->go();
大家有了解過(guò)工廠模式應(yīng)該都知道,工廠模式有三種,那就是一般工廠模式(靜態(tài)工廠模式),工廠模式,還有就是抽象工廠模式,咱這里并未把所有的案例全部介紹完畢,不過(guò)嘞,咱們可以跟著網(wǎng)上的一個(gè)案例,來(lái)簡(jiǎn)單了解下工廠模式的三種變形的過(guò)程。
首先,我們來(lái)假設(shè)有個(gè)關(guān)于個(gè)人事務(wù)管理的項(xiàng)目,功能之一就是管理Appointment(預(yù)約)對(duì)象。我們的業(yè)務(wù)團(tuán)隊(duì)和A公司建立了關(guān)系,目前需要使用一個(gè)叫做BloggsCal格式來(lái)和他們交流預(yù)約相關(guān)的數(shù)據(jù),但是業(yè)務(wù)部門提醒可能會(huì)有更多的數(shù)據(jù)格式,所以解碼器可能會(huì)有多種,我們呢,為了避免在邏輯代碼中使用過(guò)多的if else,可能就會(huì)需要使用工廠模式來(lái)將創(chuàng)造者和使用者分開(kāi)。
那么,我們就需要兩個(gè)類,一個(gè)類AppEncoder用于定義一個(gè)解碼器,將A公司傳來(lái)的數(shù)據(jù)解碼;另外一個(gè)類CommsManager用于獲取該解碼器,就是調(diào)用AppEncoder類,用于與A公司進(jìn)行通信。使用模式術(shù)語(yǔ)說(shuō),CommsManager就是創(chuàng)造者,AppEncoder就是產(chǎn)品(一個(gè)創(chuàng)造者、一個(gè)產(chǎn)品,將類的實(shí)例化和對(duì)象的使用分離開(kāi),這就是工廠模式的思想)。
咱們先來(lái)通過(guò)簡(jiǎn)單工廠模式實(shí)現(xiàn)上述任務(wù)場(chǎng)景,如下:
//產(chǎn)品類 class BloggsApptEncoder { function encode() { return "Appointment data encoded in BloggsCal format\n"; } } //創(chuàng)造者類 class CommsManager { function static getBloggsApptEncoder() { return new BloggsApptEncoder(); } }
大概明白了奧,好啦,現(xiàn)在又有新任務(wù)了,業(yè)務(wù)部門告訴我們需要新增一種數(shù)據(jù)格式MegCal,來(lái)完成數(shù)據(jù)交流,那么我們就需要新增對(duì)應(yīng)的解碼器類,然后直接在commsManager新增參數(shù)來(lái)標(biāo)識(shí)需要實(shí)例化哪個(gè)解碼器,如下:
class CommsManager { const BLOGGS = 1; const MEGA = 2; private $mode; public function __construct( $mode ) { $this->mode = $mode; } function getApptEncoder() { switch($this->mode) { case (self::MEGA): return new MegaApptEncoder(); default: return new BloggsApptEncoder(); } } }
上述兩個(gè)案例綜合起來(lái)就是簡(jiǎn)單工廠模式了,它符合現(xiàn)實(shí)中的情況,而且客戶端免除了直接創(chuàng)建產(chǎn)品對(duì)象的責(zé)任,而僅僅負(fù)責(zé)“消費(fèi)”產(chǎn)品(正如暴發(fā)戶所為)。
接下來(lái),我們從開(kāi)閉原則上來(lái)分析下簡(jiǎn)單工廠模式,當(dāng)新增一種數(shù)據(jù)格式的時(shí)候,只要符合抽象產(chǎn)品格式,那么只要通知工廠類知道就可以被使用了(即創(chuàng)建一個(gè)新的解碼器類,繼承抽象解碼器ApptEncoder),那么對(duì)于產(chǎn)品部分來(lái)說(shuō),它是符合開(kāi)閉原則的——對(duì)擴(kuò)展開(kāi)放、對(duì)修改關(guān)閉,但是對(duì)于工廠類不太理想,因?yàn)槊吭黾右桓鞲袷剑家诠S類中增加相應(yīng)的商業(yè)邏輯和判斷邏輯,這顯自然是違背開(kāi)閉原則的。
然而在實(shí)際應(yīng)用中,很可能產(chǎn)品是一個(gè)多層次的樹(shù)狀結(jié)構(gòu),這時(shí)候由于簡(jiǎn)單工廠模式中只有一個(gè)工廠類來(lái)對(duì)應(yīng)這些產(chǎn)品,所以這可能會(huì)把我們的上帝累壞了,因此簡(jiǎn)單工廠模式只適用于業(yè)務(wù)簡(jiǎn)單的情況下或者具體產(chǎn)品很少增加的情況,而對(duì)于復(fù)雜的業(yè)務(wù)環(huán)境可能不太適應(yīng)了,這個(gè)時(shí)候就應(yīng)該由工廠方法模式來(lái)出場(chǎng)了。
又來(lái)新需求了,那就是每種格式的預(yù)約數(shù)據(jù)中,需要提供頁(yè)眉和頁(yè)腳來(lái)描述每次預(yù)約。
咱們先來(lái)用簡(jiǎn)單工廠模式來(lái)實(shí)現(xiàn)上述功能,如下:
// 簡(jiǎn)單工廠模式 class CommsManager { const BLOGGS = 1; const MEGA = 2; private = $mode; public function __construct( $mode ) { $this->mode = $mode; } // 生成解碼器對(duì)應(yīng)的頁(yè)眉 public function getHeaderText() { switch( $this->mode ) { case ( self::MEGA ): return "MegaCal header\n"; default: return "BloggsCal header\n"; } } // 生成解碼器 public function getApptEncoder() { switch( $this->mode ) { case ( self::MEGA ): return new MegaApptEncoder(); default: return new BloggsApptEncoder();; } } }
從上述代碼中,我們可以看到,相同的條件語(yǔ)句switch在不同的方法中出現(xiàn)了重復(fù),而且如果添加新的數(shù)據(jù)格式,那么需要改動(dòng)的類過(guò)多。所以需要對(duì)我們的結(jié)構(gòu)進(jìn)行修改,以求更容易擴(kuò)展和維護(hù)。
我們可以使用創(chuàng)造者子類分別生成對(duì)應(yīng)的產(chǎn)品,這樣添加新的數(shù)據(jù)格式時(shí),只需要添加一個(gè)創(chuàng)造者子類即可,方便擴(kuò)展和維護(hù),這也就是比簡(jiǎn)單工廠模式更復(fù)雜一點(diǎn)的工廠模式,如下:
// 工廠模式 abstract class CommsManager { abstract function getHeaderText(); abstract function getApptEncoder(); abstract function getFooterText(); } class BloggsCommsManager extends CommsManager { function getHeaderText() { return "BloggsCal Header\n"; } function getApptEncoder() { return new BloggsApptEncoder(); } function getFooterText() { return "BloggsCal Footer\n"; } } class MegaCommsManager extends CommsManager { function getHeaderText() { return "MegaCal Header\n"; } function getApptEncoder() { return new MegaApptEncoder(); } function getFooterText() { return "MegaCal Footer\n"; } }
在這個(gè)時(shí)候,如果有新的數(shù)據(jù)格式,只需要添加一個(gè)創(chuàng)造類的子類即可,例如此時(shí)想獲取MegaCal對(duì)應(yīng)的解碼器直接通過(guò)MegaCommsManager::getApptEncoder()
獲取即可。
完了么?我只能說(shuō),這個(gè)實(shí)例還沒(méi)有完事。
新需求又來(lái)了,那就是不僅需要和A公司交流預(yù)約數(shù)據(jù)(Appointment),還需要交流待辦事宜(Ttd)、聯(lián)系人(Contact)等數(shù)據(jù),同樣的這些數(shù)據(jù)交流的格式也是BloggsCal和MegaCal,這個(gè)時(shí)候,我們可以直接在對(duì)應(yīng)解碼器的子類中添加處理事宜(TtD)和聯(lián)系人(Contact)的方法,如下:
// 抽象工廠模式 abstract class CommsManager { abstract function getHeaderText(); abstract function getApptEncoder(); abstract function getTtdEncoder(); abstract function getContactEncoder(); abstract function getFooterText(); } class BloggsCommsManager extends CommsManager { function getHeaderText() { return "BloggsCal Header\n"; } function getApptEncoder() { return new BloggsApptEncoder(); } function getTtdEncoder() { return new BloggsTtdEncoder(); } function getContactEncoder() { return new BloggsContactEncoder(); } function getFooterText() { return "BloggsCal Footer\n"; } } class MegaCommsManager extends CommsManager { function getHeaderText() { return "MegaCal Header\n"; } function getApptEncoder() { return new MegaApptEncoder(); } function getTtdEncoder() { return new MegaTtdEncoder(); } function getContactEncoder() { return new MegaContactEncoder(); } function getFooterText() { return "MegaCal Footer\n"; } } //當(dāng)然需要添加對(duì)應(yīng)的TtdEncoder抽象類和ContactEncoder抽象類,以及他們的子類。
好啦,到這里就算是差不多結(jié)束了工廠模式的變形過(guò)程,我們可以來(lái)簡(jiǎn)單歸納下這個(gè)變形過(guò)程中的核心,如下:
1.將系統(tǒng)和實(shí)現(xiàn)的細(xì)節(jié)分離開(kāi),我們可在示例中移除或者添加任意數(shù)目的編碼格式而不會(huì)影響系統(tǒng)。
2.對(duì)系統(tǒng)中功能相關(guān)的元素強(qiáng)制進(jìn)行組合,因此,通過(guò)使用BloggsCommsManager,可以確定只使用與BloggsCal有關(guān)的類。
3.添加新產(chǎn)品時(shí)將會(huì)令人苦惱,因?yàn)椴粌H需要?jiǎng)?chuàng)建新產(chǎn)品的具體實(shí)現(xiàn),而且為了支持它,我們必須修改抽象創(chuàng)建者和它的每個(gè)具體實(shí)現(xiàn)。
當(dāng)然,我們可以創(chuàng)建一個(gè)標(biāo)志參數(shù)來(lái)決定返回什么對(duì)象的單一的make()方法,而不用給每個(gè)工廠創(chuàng)建獨(dú)立的方法,如下:
abstract class CommsManager { const APPT = 1; const TTD = 2; const CONTACT = 3; abstract function getHeaderText(); abstract function make ( $flag_init ); abstract function getFooterText(); } class BloggsCommsManager extends CommsManager { function getHeaderText() { return "BloggsCal Header\n"; } function make( $flag_init ) { switch ($flag_init) { case self::APPT: return new BloggsApptEncoder(); case self::TTD: return new BloggsTtdEncoder(); case self::CONTACT: return new BloggsContactEncoder(); } } function getFooterText() { return "BloggsCal Header\n"; } }
此時(shí),如果還需要添加交流其它的數(shù)據(jù),此時(shí)只需在抽象創(chuàng)造者中添加一個(gè)新的flag_init標(biāo)識(shí),并在子創(chuàng)造者中的make方法中添加一個(gè)條件,相比來(lái)說(shuō)比原先的更加容易擴(kuò)展,只需修改少數(shù)地方即可。
最后,來(lái)簡(jiǎn)單總結(jié)下:
簡(jiǎn)單工廠:適用于生成數(shù)量少,功能簡(jiǎn)單的產(chǎn)品(BloggApptEncoder和MegaApptEncoder)
工廠模式:適用于生成數(shù)量多,功能復(fù)雜的產(chǎn)品(多個(gè)產(chǎn)品樹(shù)[BloggCal,MegaCal]、單個(gè)產(chǎn)品族[apptEncoder]),相比簡(jiǎn)單工廠來(lái)說(shuō):業(yè)務(wù)更復(fù)雜,功能更多,但是產(chǎn)品族還是單個(gè)。
抽象工廠:適用于生成多個(gè)產(chǎn)品族、多個(gè)產(chǎn)品樹(shù)的情景(產(chǎn)品族[appt,ttd,contact],產(chǎn)品樹(shù)[Bloggcal,megaCal])。相比于工廠模式,更容易擴(kuò)展添加新的產(chǎn)品族
好啦,本次記錄就到這里了。
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《PHP基本語(yǔ)法入門教程》、《PHP運(yùn)算與運(yùn)算符用法總結(jié)》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫(kù)操作入門教程》及《php常見(jiàn)數(shù)據(jù)庫(kù)操作技巧匯總》
希望本文所述對(duì)大家PHP程序設(shè)計(jì)有所幫助。
上一篇:PHP實(shí)用小技巧之調(diào)用錄像的方法
欄 目:PHP編程
本文標(biāo)題:PHP設(shè)計(jì)模式之工廠模式(Factory)入門與應(yīng)用詳解
本文地址:http://mengdiqiu.com.cn/a1/PHPbiancheng/11055.html
您可能感興趣的文章
- 04-02關(guān)于txt數(shù)據(jù)庫(kù)php的信息
- 04-02php本站才可以請(qǐng)求數(shù)據(jù) php本地?cái)?shù)據(jù)庫(kù)
- 04-02網(wǎng)頁(yè)里php操作數(shù)據(jù)庫(kù) php網(wǎng)頁(yè)例子
- 04-02php打印請(qǐng)求數(shù)據(jù) php打印輸出結(jié)果
- 04-02php數(shù)據(jù)庫(kù)地址 phpstudy 數(shù)據(jù)庫(kù)
- 04-02php插入數(shù)據(jù)庫(kù)為亂碼 php連接數(shù)據(jù)庫(kù)亂碼
- 04-02php數(shù)據(jù)庫(kù)數(shù)據(jù)相加 php數(shù)據(jù)庫(kù)添加數(shù)據(jù)語(yǔ)句
- 04-02php數(shù)據(jù)庫(kù)輸入變量 php里輸出數(shù)據(jù)庫(kù)數(shù)據(jù)函數(shù)
- 04-02數(shù)據(jù)權(quán)限架構(gòu)思路php 數(shù)據(jù)權(quán)限設(shè)計(jì)方案
- 04-02php如何用導(dǎo)入數(shù)據(jù) php用來(lái)導(dǎo)入其他文件的語(yǔ)句


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02php本站才可以請(qǐng)求數(shù)據(jù) php本地?cái)?shù)據(jù)庫(kù)
- 04-02關(guān)于txt數(shù)據(jù)庫(kù)php的信息
- 04-02php打印請(qǐng)求數(shù)據(jù) php打印輸出結(jié)果
- 04-02網(wǎng)頁(yè)里php操作數(shù)據(jù)庫(kù) php網(wǎng)頁(yè)例子
- 04-02php插入數(shù)據(jù)庫(kù)為亂碼 php連接數(shù)據(jù)庫(kù)亂
- 04-02php數(shù)據(jù)庫(kù)地址 phpstudy 數(shù)據(jù)庫(kù)
- 04-02php數(shù)據(jù)庫(kù)數(shù)據(jù)相加 php數(shù)據(jù)庫(kù)添加數(shù)據(jù)
- 04-02數(shù)據(jù)權(quán)限架構(gòu)思路php 數(shù)據(jù)權(quán)限設(shè)計(jì)方
- 04-02php數(shù)據(jù)庫(kù)輸入變量 php里輸出數(shù)據(jù)庫(kù)數(shù)
- 04-02php如何用導(dǎo)入數(shù)據(jù) php用來(lái)導(dǎo)入其他文
隨機(jī)閱讀
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10C#中split用法實(shí)例總結(jié)
- 04-02jquery與jsp,用jquery
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-11Mac OSX 打開(kāi)原生自帶讀寫NTFS功能(圖文
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10delphi制作wav文件的方法