概述
抽象工廠模式是對(duì)象的創(chuàng)建模式案腺,它是工廠方法模式的進(jìn)一步推廣庆冕。
假設(shè)一個(gè)子系統(tǒng)需要一些產(chǎn)品對(duì)象,而這些產(chǎn)品又屬于一個(gè)以上的產(chǎn)品等級(jí)結(jié)構(gòu)劈榨。那么為了將消費(fèi)這些產(chǎn)品對(duì)象的責(zé)任和創(chuàng)建這些產(chǎn)品對(duì)象的責(zé)任分割開(kāi)來(lái)访递,可以引進(jìn)抽象工廠模式。這樣的話同辣,消費(fèi)產(chǎn)品的一方不需要直接參與產(chǎn)品的創(chuàng)建工作拷姿,而只需要向一個(gè)公用的工廠接口請(qǐng)求所需要的產(chǎn)品。
通過(guò)使用抽象工廠模式旱函,可以處理具有相同(或者相似)等級(jí)結(jié)構(gòu)中的多個(gè)產(chǎn)品族中的產(chǎn)品對(duì)象的創(chuàng)建問(wèn)題响巢。如下圖所示:
根據(jù)產(chǎn)品角色的結(jié)構(gòu)圖,就不難給出工廠角色的結(jié)構(gòu)設(shè)計(jì)圖棒妨。
可以看出踪古,每一個(gè)工廠角色都有兩個(gè)工廠方法,分別負(fù)責(zé)創(chuàng)建分屬不同產(chǎn)品等級(jí)結(jié)構(gòu)的產(chǎn)品對(duì)象券腔。
抽象工廠的功能是為一系列相關(guān)對(duì)象或相互依賴的對(duì)象創(chuàng)建一個(gè)接口伏穆。一定要注意,這個(gè)接口內(nèi)的方法不是任意堆砌的纷纫,而是一系列相關(guān)或相互依賴的方法枕扫。比如上面例子中的主板和CPU,都是為了組裝一臺(tái)電腦的相關(guān)對(duì)象辱魁。不同的裝機(jī)方案烟瞧,代表一種具體的電腦系列诗鸭。
由于抽象工廠定義的一系列對(duì)象通常是相關(guān)或相互依賴的,這些產(chǎn)品對(duì)象就構(gòu)成了一個(gè)產(chǎn)品族参滴,也就是抽象工廠定義了一個(gè)產(chǎn)品族强岸。
這就帶來(lái)非常大的靈活性,切換產(chǎn)品族的時(shí)候卵洗,只要提供不同的抽象工廠實(shí)現(xiàn)就可以了请唱,也就是說(shuō)現(xiàn)在是以一個(gè)產(chǎn)品族作為一個(gè)整體被切換。
核心
先上一張圖:
我們還是以苦逼的程序猿為例來(lái)說(shuō)抽象工廠模式的一些核心概念过蹂。通過(guò)上圖你可以發(fā)現(xiàn)十绑,橫縱二維坐標(biāo)可以確定平面上一個(gè)唯一的點(diǎn),這也就是抽象工廠的核心酷勺。
產(chǎn)品等級(jí)結(jié)構(gòu):就是繼承結(jié)構(gòu)本橙。就像上面Android,IOS脆诉,PHP這些技能繼承自一個(gè)抽象的技能類(譬如前面的ICode)甚亭,這個(gè)抽象類與這些子類構(gòu)成了產(chǎn)品等級(jí)結(jié)構(gòu)。 同理的Android書击胜,C語(yǔ)言書亏狰,腳本書繼承自一個(gè)工具書類,這個(gè)工具書抽象類與這些子類構(gòu)成了等級(jí)結(jié)構(gòu)偶摔。
產(chǎn)品族:抽象工廠模式中的產(chǎn)品族官方定義是指由同一個(gè)工廠生產(chǎn)的暇唾,位于不同產(chǎn)品等級(jí)結(jié)構(gòu)中的一組產(chǎn)品。 譬如上面的Android位于技能等級(jí)結(jié)構(gòu)中辰斋,Android書位于工具書等級(jí)結(jié)構(gòu)中策州,Android技能和Android書是位于不同產(chǎn)品結(jié)構(gòu)的一組產(chǎn)品,但是任何一個(gè)程序猿都需要具備技能和工具書宫仗, 譬如一個(gè)Android程序猿需要有Android技能及Android書够挂,所以這個(gè)Android程序猿就是一個(gè)產(chǎn)品族。
概念: 提供一個(gè)創(chuàng)建一系列相關(guān)或相互依賴對(duì)象的接口藕夫,而無(wú)須指定它們具體的類孽糖。抽象工廠模式又稱為Kit模式,它是一種對(duì)象創(chuàng)建型模式毅贮。
重點(diǎn): 抽象工廠模式結(jié)構(gòu)重要核心模塊:
抽象工廠:
聲明一組用于創(chuàng)建一族產(chǎn)品的方法梭姓,每一個(gè)方法對(duì)應(yīng)一種產(chǎn)品。
具體工廠:
實(shí)現(xiàn)了在抽象工廠中聲明的創(chuàng)建產(chǎn)品的方法嫩码,生成一組具體產(chǎn)品,這些產(chǎn)品構(gòu)成了一個(gè)產(chǎn)品族罪既,每一個(gè)產(chǎn)品都位于某個(gè)產(chǎn)品等級(jí)結(jié)構(gòu)中铸题。
抽象產(chǎn)品:
它為每種產(chǎn)品聲明接口铡恕,在抽象產(chǎn)品中聲明了產(chǎn)品所具有的業(yè)務(wù)方法。
具體產(chǎn)品:
定義具體工廠生產(chǎn)的具體產(chǎn)品對(duì)象丢间,實(shí)現(xiàn)抽象產(chǎn)品接口中聲明的業(yè)務(wù)方法探熔。
使用場(chǎng)景:
當(dāng)需要?jiǎng)?chuàng)建的對(duì)象是一系列相互關(guān)聯(lián)或相互依賴的產(chǎn)品族時(shí),便可以使用抽象工廠模式烘挫。 大白話意思就是一個(gè)繼承體系中诀艰,如果存在著多個(gè)等級(jí)結(jié)構(gòu)(即存在著多個(gè)抽象類,像上面的技能與工具書)饮六, 并且分屬各個(gè)等級(jí)結(jié)構(gòu)中的實(shí)現(xiàn)類之間存在著一定的關(guān)聯(lián)或者約束其垄,就可以使用抽象工廠模式。當(dāng)然了卤橄, 同樣的道理就是如果各個(gè)等級(jí)結(jié)構(gòu)中的實(shí)現(xiàn)類之間不存在關(guān)聯(lián)或約束绿满,則使用多個(gè)獨(dú)立的工廠來(lái)對(duì)產(chǎn)品進(jìn)行創(chuàng)建。
程序?qū)嵗?/p>
如下實(shí)例就是上圖何如上文字解釋的實(shí)現(xiàn)代碼窟扑,具體不再解釋:
復(fù)制代碼代碼如下:
package yanbober.github.io;
/*技能等級(jí)結(jié)構(gòu)部分*/
interface ICode {
void coding();
}
class CodeImplAndroid implements ICode {
@Override
public void coding() {
System.out.println("Coding Android!");
}
}
class CodeImplPHP implements ICode {
@Override
public void coding() {
System.out.println("Coding PHP!");
}
}
/*工具書等級(jí)結(jié)構(gòu)*/
interface INeedBook {
void lookBook();
}
class NeedBookImplAndroid implements INeedBook {
@Override
public void lookBook() {
System.out.println("Look Android Book!");
}
}
class NeedBookImplPHP implements INeedBook {
@Override
public void lookBook() {
System.out.println("Look PHP Book!");
}
}
/*產(chǎn)品族*/
interface IAbstractFactory {
ICode getCodingSkill();
INeedBook getNeedBook();
}
class FactoryImplAndroid implements IAbstractFactory {
@Override
public ICode getCodingSkill() {
return new CodeImplAndroid();
}
@Override
public INeedBook getNeedBook() {
return new NeedBookImplAndroid();
}
}
class FactoryImplPHP implements IAbstractFactory {
@Override
public ICode getCodingSkill() {
return new CodeImplPHP();
}
@Override
public INeedBook getNeedBook() {
return new NeedBookImplPHP();
}
}
public class Main {
public static void main(String[] args) {
IAbstractFactory factory = new FactoryImplAndroid();
ICode code = factory.getCodingSkill();
INeedBook book = factory.getNeedBook();
code.coding();
book.lookBook();
factory = new FactoryImplPHP();
code = factory.getCodingSkill();
book = factory.getNeedBook();
code.coding();
book.lookBook();
}
}
技巧Tips:依舊可以使用配置與反射實(shí)現(xiàn)自動(dòng)適應(yīng)喇颁。
總結(jié)一把
抽象工廠模式的優(yōu)點(diǎn):
和前面一樣,隔離具體類的生成嚎货,使客戶并不需要知道什么被創(chuàng)建橘霎。
增加新的產(chǎn)品族很方便,無(wú)須修改已有系統(tǒng)殖属,符合“開(kāi)閉原則”姐叁。
抽象工廠模式的缺點(diǎn):
增加新的產(chǎn)品等級(jí)結(jié)構(gòu)麻煩,需要對(duì)原有系統(tǒng)進(jìn)行較大的修改忱辅,甚至需要修改抽象層代碼七蜘,違背“開(kāi)閉原則”