這篇我們來講抽象工廠模式演怎。上一篇提到工廠方法是抽象工廠的組成部分粉怕,那么我們先來看下什么是抽象工廠:
根據(jù)Wesley(1994)的定義吼句,抽象工廠提供創(chuàng)建一系列相關(guān)或相互依賴對象的接口根资,并且無需指定他們具體的類费坊。
這樣可能還是比較抽象倒槐,我們先來看下工廠方法和抽象工廠的異同點,再舉例解釋附井。這兩個模式在很多方面都十分相似讨越,很容易讓人搞不清什么時候該用哪一個。但這兩個模式有一個共同點永毅,或者說是共同目的:創(chuàng)建對象而不讓客戶端知曉返回了什么確切的具體對象把跨。下面是它們兩個的對比:
抽象工廠 | 工廠方法 |
---|---|
通過對象組合來創(chuàng)建抽象產(chǎn)品 | 通過類繼承來創(chuàng)建抽象產(chǎn)品 |
創(chuàng)建多系列產(chǎn)品 | 創(chuàng)建一種產(chǎn)品 |
必須修改父類接口才能創(chuàng)建新產(chǎn)品 | 增加子類即可增加新產(chǎn)品 |
首先看第一條,抽象工廠是通過對象組合來創(chuàng)建產(chǎn)品沼死,而工廠方法是通過繼承來創(chuàng)建着逐。這是什么意思呢?我們來看下面一幅圖:
![Abstarct Factory](http://harry.iceloof.com/blog/15/abstract fac.png)
這是抽象工廠的結(jié)構(gòu)圖(不是UML,一般的結(jié)構(gòu)圖)耸别,它說明了抽象工廠模式中各個部分的關(guān)系健芭。在抽象工廠模式中,頂層是一個抽象工廠,具體來說可以是接口秀姐,可以是抽象類吟榴。它下面是具體實現(xiàn)或繼承的子類工廠,也稱為具體工廠囊扳,在這每個具體工廠里面都實現(xiàn)了抽象工廠里定義的方法,例如createA 或 createB等等兜看。每個具體工廠可以創(chuàng)建多個具體產(chǎn)品锥咸。宏觀來看,這個模式是產(chǎn)生產(chǎn)品簇的细移,具體類1和具體類2都要實現(xiàn)createA或createB方法搏予,但是怎么實現(xiàn)卻是不同的(所以圖中顏色不同),也意味兩個具體工廠類產(chǎn)生的productA和productB也不同弧轧。例如抽象工廠是一個生產(chǎn)汽車的工廠雪侥,旗下有不同的品牌,每個品牌就是一個具體工廠精绎。這些工廠需要生產(chǎn)轎車速缨,貨車和客車這3種車型(對應(yīng)createA、createB和createC)代乃,最終生產(chǎn)出來的就是具體的產(chǎn)品旬牲,但是生產(chǎn)出來的樣子肯定是不同的,發(fā)動機也是不同的等等搁吓。通過這個例子原茅,相信您的更進一步理解表中的"組合"和"系列"等字眼。
細(xì)心地讀者會發(fā)現(xiàn)堕仔,在抽象工廠中擂橘,頂層接口或抽象類定義了標(biāo)準(zhǔn),例如生產(chǎn)什么摩骨,具體的方法接口通贞。這樣的話,要是增加一條產(chǎn)品線仿吞,或者說是一個品牌是很容易的滑频,就是多加一個具體工廠類即可。但是唤冈,如果要是多加一個產(chǎn)品的話峡迷,改動很大,需要從頂?shù)较滦薷模婕八幸阎惢娓恪_@時候可以考慮下工廠方法模式彤避。
工廠方法模式適用與多個簡單產(chǎn)品的創(chuàng)建,它們都位于同一層級中夯辖。頂層同樣是一個抽象工廠琉预,但是和真正的抽象工廠不同的是,在工廠方法中的頂層類代表一個抽象的產(chǎn)品蒿褂,也就是說其本身就是一個產(chǎn)品圆米,只不過被抽象出來了,它的子類進一步細(xì)化它啄栓,并返回特定的產(chǎn)品娄帖,大多情況下,其子類就返回自己本身昙楚,因為它本身就是一個具體的產(chǎn)品類近速。舉個簡單的例子,在項目中堪旧,我要給HTTP服務(wù)器發(fā)送很多request削葱,例如login request,register request淳梦,authentication request等等析砸。它們其實都是具體的產(chǎn)品類,繼承自HTTPRequest這個"抽象工廠"爆袍,當(dāng)我要添加里一個"產(chǎn)品"時干厚,我只需要繼承HTTPRequest,之后override一些方法即可螃宙。
書中的例子
下圖是《編程之道》中的例子:
圖中有兩個品牌, Sierra和Acme蛮瞄,每個品牌都有一個具體工廠去生產(chǎn)具體的產(chǎn)品。不過在這個例子里面谆扎,這些view, button, 和toolbar更像是零件挂捅,之后需要在客戶端組裝。這就是典型的抽象工廠模式: 生產(chǎn)一系列產(chǎn)品堂湖。具體的代碼我就不貼了闲先,因為真的想一下就知道里面是什么(返回不同的類實例)。
Cocoa Touch框架中的抽象工廠
在我們平常用的類中也有很多是基于抽象工廠模式的无蜂,例如NSNumber伺糠,NSData,NSArray斥季,NSString和NSDictionary训桶。這些其實都是"類簇"的概念累驮,例如NSNumer有NSCFNumber和NSCFBoolean兩種具體子類(具體工廠);NSString也有NSCFString等舵揭。這些都是基于抽像工廠的基本概念而延伸出來的谤专,而暴露給客戶端的僅僅是頂層的抽像類,封裝了其余的所有細(xì)節(jié)午绳。如果您有興趣探究這些置侍,推薦閱讀《Objective-C編程之道》.