工廠模式

來源:https://www.cnblogs.com/zailushang1996/p/8601808.html
一、簡單工廠模式

一個栗子:

我喜歡吃面條,抽象一個面條基類,(接口也可以),這是產品的抽象類。

publicabstractclass INoodles {

? ? /**? ? * 描述每種面條啥樣的

? ? */publicabstractvoid desc();

}

先來一份蘭州拉面(具體的產品類):

publicclassLzNoodlesextends INoodles {

? ? @Override

? ? publicvoid desc() {

? ? ? ? System.out.println("蘭州拉面 上海的好貴 家里才5 6塊錢一碗");

? ? }

}

程序員加班必備也要吃泡面(具體的產品類):

publicclassPaoNoodlesextends INoodles {

? ? @Override

? ? publicvoid desc() {

? ? ? ? System.out.println("泡面好吃 可不要貪杯");

? ? }

}

還有我最愛吃的家鄉的干扣面(具體的產品類):


publicclassGankouNoodlesextends INoodles {

? ? @Override

? ? publicvoid desc() {

? ? ? ? System.out.println("還是家里的干扣面好吃 6塊一碗");

? ? }

}

準備工作做完了,我們來到一家“簡單面館”(簡單工廠類),菜單如下:

publicclass SimpleNoodlesFactory {

? ? publicstaticfinalintTYPE_LZ = 1;//蘭州拉面

publicstaticfinalintTYPE_PM = 2;//泡面

publicstaticfinalintTYPE_GK = 3;//干扣面

publicstaticINoodles createNoodles(int type) {

? ? ? ? switch (type) {

? ? ? ? ? ? case TYPE_LZ:

? ? ? ? ? ? ? ? returnnew LzNoodles();

? ? ? ? ? ? case TYPE_PM:

? ? ? ? ? ? ? ? returnnew PaoNoodles();

? ? ? ? ? ? case TYPE_GK:

? ? ? ? ? ? default:

? ? ? ? ? ? ? ? returnnew GankouNoodles();

? ? ? ? }

? ? }

}

簡單面館就提供三種面條(產品),你說你要啥,他就給你啥。這里我點了一份干扣面:

/** * 簡單工廠模式

*/ INoodles noodles = SimpleNoodlesFactory.createNoodles(SimpleNoodlesFactory.TYPE_GK);

noodles.desc();

輸出:

還是家里的干扣面好吃 6塊一碗

特點

1 它是一個具體的類,非接口 抽象類。有一個重要的create()方法,利用if或者 switch創建產品并返回。

2 create()方法通常是靜態的,所以也稱之為靜態工廠。

缺點

1 擴展性差(我想增加一種面條,除了新增一個面條產品類,還需要修改工廠類方法)

2 不同的產品需要不同額外參數的時候 不支持。


二、工廠方法模式

1.模式描述

提供一個用于創建對象的接口(工廠接口),讓其實現類(工廠實現類)決定實例化哪一個類(產品類),并且由該實現類創建對應類的實例。

2.模式作用

可以一定程度上解耦,消費者和產品實現類隔離開,只依賴產品接口(抽象產品),產品實現類如何改動與消費者完全無關。

可以一定程度增加擴展性,若增加一個產品實現,只需要實現產品接口,修改工廠創建產品的方法,消費者可以無感知(若消費者不關心具體產品是什么的情況)。

可以一定程度增加代碼的封裝性、可讀性。清楚的代碼結構,對于消費者來說很少的代碼量就可以完成很多工作。

等等。//TODO

另外,抽象工廠才是實際意義的工廠模式,工廠方法只是抽象工廠的一個比較常見的情況。

3.適用場景

消費者不關心它所要創建對象的類(產品類)的時候。

消費者知道它所要創建對象的類(產品類),但不關心如何創建的時候。

等等。//TODO

例如:hibernate里通過sessionFactory創建session、通過代理方式生成ws客戶端時,通過工廠構建報文中格式化數據的對象。

4.模式要素

提供一個產品類的接口。產品類均要實現這個接口(也可以是abstract類,即抽象產品)。

提供一個工廠類的接口。工廠類均要實現這個接口(即抽象工廠)。

由工廠實現類創建產品類的實例。工廠實現類應有一個方法,用來實例化產品類。

5.類圖


6.模式實例代碼

工廠:

package com.demoFound.factoryMethod.factory;?

? import com.demoFound.factoryMethod.message.IMyMessage;?

? /**

* 工廠方法模式_工廠接口

*?

* @author popkidorc

*?

*/publicinterface IMyMessageFactory {?


? ? public IMyMessage createMessage(String messageType);?

}?

package com.demoFound.factoryMethod.factory;?

? import java.util.HashMap;? import java.util.Map;?

? import com.demoFound.factoryMethod.message.IMyMessage;? import com.demoFound.factoryMethod.message.MyMessageEmail;? import com.demoFound.factoryMethod.message.MyMessageOaTodo;? import com.demoFound.factoryMethod.message.MyMessageSms;?

? /**

* 工廠方法模式_工廠實現

*?

* @author popkidorc

*?

*/publicclassMyMessageFactoryimplements IMyMessageFactory {?


? ? @Override?

? ? public IMyMessage createMessage(String messageType) {?

? ? ? ? // 這里的方式是:消費者知道自己想要什么產品;若生產何種產品完全由工廠決定,則這里不應該傳入控制生產的參數。? ? ? ? ? IMyMessage myMessage;?

? ? ? ? Map messageParam =newHashMap();?

? ? ? ? // 根據某些條件去選擇究竟創建哪一個具體的實現對象,條件可以傳入的,也可以從其它途徑獲取。?

? ? ? ? // sms? if("SMS".equals(messageType)) {?

? ? ? ? ? ? myMessage =new MyMessageSms();?

? ? ? ? ? ? messageParam.put("PHONENUM", "123456789");?

? ? ? ? } else// OA待辦? if("OA".equals(messageType)) {?

? ? ? ? ? ? myMessage =new MyMessageOaTodo();?

? ? ? ? ? ? messageParam.put("OAUSERNAME", "testUser");?

? ? ? ? } else// email? if("EMAIL".equals(messageType)) {?

? ? ? ? ? ? myMessage =new MyMessageEmail();?

? ? ? ? ? ? messageParam.put("EMAIL", "test@test.com");?

? ? ? ? } else// 默認生產email這個產品? ? ? ? ? {?

? ? ? ? ? ? myMessage =new MyMessageEmail();?

? ? ? ? ? ? messageParam.put("EMAIL", "test@test.com");?

? ? ? ? }?

? ? ? ? myMessage.setMessageParam(messageParam);?

? ? ? ? return myMessage;?

? ? }?

}

產品:

package com.demoFound.factoryMethod.message;?

? import java.util.Map;?

? /**

* 工廠方法模式_產品接口

*?

* @author popkidorc

*?

*/publicinterface IMyMessage {?


? ? publicMap getMessageParam();?


? ? publicvoidsetMessageParam(Map messageParam);?


? ? publicvoidsendMesage()throwsException;// 發送通知/消息? ?

}?


package com.demoFound.factoryMethod.message;?

? import java.util.Map;?

? /**

* 工廠方法模式_虛擬產品類

*?

* @author popkidorc

*?

*/publicabstractclassMyAbstractMessageimplements IMyMessage {?


? ? privateMap messageParam;// 這里可以理解為生產產品所需要的原材料庫。最好是個自定義的對象,這里為了不引起誤解使用Map。? ?

? ? @Override?

? ? publicMap getMessageParam() {?

? ? ? ? return messageParam;?

? ? }?


? ? @Override?

? ? publicvoidsetMessageParam(Map messageParam) {?

? ? ? ? this.messageParam = messageParam;?

? ? }?

}?

package com.demoFound.factoryMethod.message;?

? /**

* 工廠方法模式_email產品

*?

* @author popkidorc

*?

*/publicclassMyMessageEmailextends MyAbstractMessage {?


? ? @Override?

? ? publicvoidsendMesage()throws Exception {?

? ? ? ? // TODO Auto-generated method stub? if(null== getMessageParam() ||null== getMessageParam().get("EMAIL")?

? ? ? ? ? ? ? ? || "".equals(getMessageParam().get("EMAIL"))) {?

? ? ? ? ? ? thrownewException("發送短信,需要傳入EMAIL參數");// 為了簡單起見異常也不自定義了? }// 另外郵件內容,以及其他各種協議參數等等都要處理? ?

? ? ? ? System.out.println("我是郵件,發送通知給" + getMessageParam().get("EMAIL"));?

? ? }?


}?


package com.demoFound.factoryMethod.message;?

? /**

* 工廠方法模式_oa待辦產品

*?

* @author popkidorc

*?

*/publicclassMyMessageOaTodoextends MyAbstractMessage {?


? ? @Override?

? ? publicvoidsendMesage()throws Exception {?

? ? ? ? // TODO Auto-generated method stub? if(null== getMessageParam()?

? ? ? ? ? ? ? ? ||null== getMessageParam().get("OAUSERNAME")?

? ? ? ? ? ? ? ? || "".equals(getMessageParam().get("OAUSERNAME"))) {?

? ? ? ? ? ? thrownewException("發送OA待辦,需要傳入OAUSERNAME參數");// 為了簡單起見異常也不自定義了? }// 這里的參數需求就比較多了不一一處理了? ?

? ? ? ? System.out?

? ? ? ? ? ? ? ? .println("我是OA待辦,發送通知給" + getMessageParam().get("OAUSERNAME"));?

? ? }?


}?

package com.demoFound.factoryMethod.message;?

? /**

* 工廠方法模式_sms產品

*?

* @author popkidorc

*?

*/publicclassMyMessageSmsextends MyAbstractMessage {?


? ? @Override?

? ? publicvoidsendMesage()throws Exception {?

? ? ? ? // TODO Auto-generated method stub? if(null== getMessageParam()?

? ? ? ? ? ? ? ? ||null== getMessageParam().get("PHONENUM")?

? ? ? ? ? ? ? ? || "".equals(getMessageParam().get("PHONENUM"))) {?

? ? ? ? ? ? thrownewException("發送短信,需要傳入PHONENUM參數");// 為了簡單起見異常也不自定義了? }// 另外短信信息,以及其他各種協議參數等等都要處理? ?

? ? ? ? System.out.println("我是短信,發送通知給" + getMessageParam().get("PHONENUM"));?

? ? }?


}

消費者:

package com.demoFound.factoryMethod;?

? import com.demoFound.factoryMethod.factory.IMyMessageFactory;? import com.demoFound.factoryMethod.factory.MyMessageFactory;? import com.demoFound.factoryMethod.message.IMyMessage;?

? /**

* 工廠方法模式_消費者類

*?

* @author popkidorc

*?

*/publicclass MyFactoryMethodMain {?


? ? publicstaticvoid main(String[] args) {?

? ? ? ? IMyMessageFactory myMessageFactory =new MyMessageFactory();?

? ? ? ? IMyMessage myMessage;?

? ? ? ? // 對于這個消費者來說,不用知道如何生產message這個產品,耦合度降低? try {?

? ? ? ? ? ? // 先來一個短信通知? myMessage = myMessageFactory.createMessage("SMS");?

? ? ? ? ? ? myMessage.sendMesage();?


? ? ? ? ? ? // 來一個oa待辦? myMessage = myMessageFactory.createMessage("OA");?

? ? ? ? ? ? myMessage.sendMesage();?


? ? ? ? ? ? // 來一個郵件通知? myMessage = myMessageFactory.createMessage("EMAIL");?

? ? ? ? ? ? myMessage.sendMesage();?

? ? ? ? } catch (Exception e) {?

? ? ? ? ? ? e.printStackTrace();?

? ? ? ? }?

? ? }?

}?



三、抽象工廠模式

定義:為創建一組相關或相互依賴的對象提供一個接口,而且無需指定他們的具體類。

類型:創建類模式

類圖:

抽象工廠模式與工廠方法模式的區別

??????? 抽象工廠模式是工廠方法模式的升級版本,他用來創建一組相關或者相互依賴的對象。他與工廠方法模式的區別就在于,工廠方法模式針對的是一個產品等級結構;而抽象工廠模式則是針對的多個產品等級結構。在編程中,通常一個產品結構,表現為一個接口或者抽象類,也就是說,工廠方法模式提供的所有產品都是衍生自同一個接口或抽象類,而抽象工廠模式所提供的產品則是衍生自不同的接口或抽象類。

??????? 在抽象工廠模式中,有一個產品族的概念:所謂的產品族,是指位于不同產品等級結構中功能相關聯的產品組成的家族。抽象工廠模式所提供的一系列產品就組成一個產品族;而工廠方法提供的一系列產品稱為一個等級結構。我們依然拿生產汽車的例子來說明他們之間的區別。

??????? 在上面的類圖中,兩廂車和三廂車稱為兩個不同的等級結構;而2.0排量車和2.4排量車則稱為兩個不同的產品族。再具體一點,2.0排量兩廂車和2.4排量兩廂車屬于同一個等級結構,2.0排量三廂車和2.4排量三廂車屬于另一個等級結構;而2.0排量兩廂車和2.0排量三廂車屬于同一個產品族,2.4排量兩廂車和2.4排量三廂車屬于另一個產品族。

??????? 明白了等級結構和產品族的概念,就理解工廠方法模式和抽象工廠模式的區別了,如果工廠的產品全部屬于同一個等級結構,則屬于工廠方法模式;如果工廠的產品來自多個等級結構,則屬于抽象工廠模式。在本例中,如果一個工廠模式提供2.0排量兩廂車和2.4排量兩廂車,那么他屬于工廠方法模式;如果一個工廠模式是提供2.4排量兩廂車和2.4排量三廂車兩個產品,那么這個工廠模式就是抽象工廠模式,因為他提供的產品是分屬兩個不同的等級結構。當然,如果一個工廠提供全部四種車型的產品,因為產品分屬兩個等級結構,他當然也屬于抽象工廠模式了。

抽象工廠模式代碼

interface IProduct1 {?

? ? publicvoid show();?

}? interface IProduct2 {?

? ? publicvoid show();?

}?

? classProduct1implements IProduct1 {?

? ? publicvoid show() {?

? ? ? ? System.out.println("這是1型產品");?

? ? }?

}? classProduct2implements IProduct2 {?

? ? publicvoid show() {?

? ? ? ? System.out.println("這是2型產品");?

? ? }?

}?

? interface IFactory {?

? ? public IProduct1 createProduct1();?

? ? public IProduct2 createProduct2();?

}? classFactoryimplements IFactory{?

? ? public IProduct1 createProduct1() {?

? ? ? ? returnnew Product1();?

? ? }?

? ? public IProduct2 createProduct2() {?

? ? ? ? returnnew Product2();?

? ? }?

}?

? publicclass Client {?

? ? publicstaticvoid main(String[] args){?

? ? ? ? IFactory factory =new Factory();?

? ? ? ? factory.createProduct1().show();?

? ? ? ? factory.createProduct2().show();?

? ? }?

}

抽象工廠模式的優點

??????? 抽象工廠模式除了具有工廠方法模式的優點外,最主要的優點就是可以在類的內部對產品族進行約束。所謂的產品族,一般或多或少的都存在一定的關聯,抽象工廠模式就可以在類內部對產品族的關聯關系進行定義和描述,而不必專門引入一個新的類來進行管理。


抽象工廠模式的缺點

?????? 產品族的擴展將是一件十分費力的事情,假如產品族中需要增加一個新的產品,則幾乎所有的工廠類都需要進行修改。所以使用抽象工廠模式時,對產品等級結構的劃分是非常重要的。


適用場景

?????? 當需要創建的對象是一系列相互關聯或相互依賴的產品族時,便可以使用抽象工廠模式。說的更明白一點,就是一個繼承體系中,如果存在著多個等級結構(即存在著多個抽象類),并且分屬各個等級結構中的實現類之間存在著一定的關聯或者約束,就可以使用抽象工廠模式。假如各個等級結構中的實現類之間不存在關聯或約束,則使用多個獨立的工廠來對產品進行創建,則更合適一點。


總結

?????? 無論是簡單工廠模式,工廠方法模式,還是抽象工廠模式,他們都屬于工廠模式,在形式和特點上也是極為相似的,他們的最終目的都是為了解耦。在使用時,我們不必去在意這個模式到底工廠方法模式還是抽象工廠模式,因為他們之間的演變常常是令人琢磨不透的。經常你會發現,明明使用的工廠方法模式,當新需求來臨,稍加修改,加入了一個新方法后,由于類中的產品構成了不同等級結構中的產品族,它就變成抽象工廠模式了;而對于抽象工廠模式,當減少一個方法使的提供的產品不再構成產品族之后,它就演變成了工廠方法模式。

?????? 所以,在使用工廠模式時,只需要關心降低耦合度的目的是否達到了。

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,825評論 6 546
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,814評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,980評論 0 384
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 64,064評論 1 319
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,779評論 6 414
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,109評論 1 330
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,099評論 3 450
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,287評論 0 291
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,799評論 1 338
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,515評論 3 361
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,750評論 1 375
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,221評論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,933評論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,327評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,667評論 1 296
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,492評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,703評論 2 380

推薦閱讀更多精彩內容

  • 工廠模式根據抽象程度的不同分為三種: 簡單工廠模式(也叫靜態工廠模式) 工廠方法模式 抽象工廠模式 優點: 可以使...
    賈里閱讀 354評論 0 1
  • 工廠模式是我們最常用的實例化對象模式了,是用工廠方法代替new操作的一種模式。通常我們所說的工廠模式是指工廠方法模...
    zfylin閱讀 1,326評論 0 7
  • 設計原則: 要依賴抽象,不要依賴具體類 目錄 本文的結構如下: 什么是抽象工廠模式 為什么要用該模式 模式的結構 ...
    w1992wishes閱讀 1,133評論 0 6
  • D8:2019/4/16《高績效教練》第8章 什么是現狀 【1.印象最深刻的部分】 1.現狀變得更清晰,目標就會更...
    Cissy_Zou閱讀 439評論 0 0
  • 看了這篇《文章的動感如何達成》,我發現如今各公眾號平臺的大多數爆款雞湯文,也是運用了文中的方法。 一般雞湯文會在開...
    莫一米閱讀 333評論 4 0