定義
工廠方法模式使用的頻率非常高膏燕,在我們?nèi)粘5拈_(kāi)發(fā)中總能見(jiàn)到它的身影暖途。其定義如下:
Define an interface for creating an object, but let subclasses decide which class to instantiate.
定義一個(gè)用于創(chuàng)建對(duì)象的接口脊凰,讓子類決定實(shí)例化哪一個(gè)類掏愁。工廠方法使用一個(gè)類的實(shí)例化延遲到其子類匿醒。
工廠方法模式的通用類圖如下:
在工廠方法模式中睡榆,抽象產(chǎn)品類Product負(fù)責(zé)定義產(chǎn)品的共性夺谁,實(shí)現(xiàn)了對(duì)事物最抽象的定義;Creator為抽象創(chuàng)建類肉微,也就是抽象工廠匾鸥,具體如何創(chuàng)建產(chǎn)品類是由具體的實(shí)現(xiàn)工廠ConcreteCreator完成的。
通用框架
工廠方法模式的變種較多碉纳,我們來(lái)看一個(gè)比較實(shí)用的通用源碼勿负。
抽象產(chǎn)品類:
public abstract class Product {
// 產(chǎn)品類的公共方法
public void bizComm() {
// 業(yè)務(wù)邏輯處理
}
// 抽象方法
public abstract void methodXXX();
}
具體產(chǎn)品類:
public class ConcreteProduct1 extends Product {
public void methodXXX() {
// 業(yè)務(wù)邏輯處理
}
}
public class ConcreteProduct2 extends Product {
public void methodXXX() {
// 業(yè)務(wù)邏輯處理
}
}
抽象工廠類
public abstract class Creator {
// 創(chuàng)建一個(gè)產(chǎn)品對(duì)象,其輸入?yún)?shù)類型可以自行設(shè)置,
// 通常為String, Enum, Class等奴愉,當(dāng)然也可以為空
public abstract <T extends Product> T createProduct(Class<T> c);
}
具體工廠類
public class ConcreteCreator extends Creator {
public <T extends Product> T createProduct(Class<T> c) {
Product product = null;
try {
product = (Product) Class.forName(c.getName()).newInstance();
} catch (Exception e) {
// 異常處理
}
return (T) product;
}
}
場(chǎng)景類的具體調(diào)用方法如下:
public class Client {
public static void main(String[] args) {
Creator creator = new ConcreteCreator();
Product product = creator.createProduct(ConcreteProduct1.class);
// 其他業(yè)務(wù)處理
}
}
優(yōu)點(diǎn)
首先琅摩,良好的封裝性,代碼結(jié)構(gòu)清晰锭硼,一個(gè)對(duì)象的創(chuàng)建是有條件約束的房资,如一個(gè)調(diào)用者需要一個(gè)具體的產(chǎn)品對(duì)象,只要知道這個(gè)產(chǎn)品的類名就可以了檀头,不用知道創(chuàng)建對(duì)象的艱辛過(guò)程轰异,降低模塊間的耦合。
其次暑始,工廠方法模式的擴(kuò)展性非常優(yōu)秀搭独,在增加產(chǎn)品類的情況下,只要適當(dāng)?shù)匦薷木唧w的工廠類或擴(kuò)展一個(gè)工廠類廊镜,就可以完成“擁抱變化”牙肝。
再次,屏蔽了產(chǎn)品類嗤朴。產(chǎn)品類的實(shí)現(xiàn)如何變化配椭,調(diào)用者都不需要關(guān)心,它只需要關(guān)心產(chǎn)品的接口雹姊,只要接口保持不變股缸,系統(tǒng)中的上層模塊就不要發(fā)生變化。
最后容为,工廠方法模式是典型的解耦框架乓序。高層模塊只需要知道產(chǎn)品的抽象類寺酪,其他實(shí)現(xiàn)類都不用關(guān)心坎背,符合迪米特法則,我不需要的就不要去交流寄雀;也符合依賴倒置原則得滤,只依賴產(chǎn)品類的抽象;當(dāng)然也符合里氏替換原則盒犹,使用產(chǎn)品子類替換產(chǎn)品父類懂更。