試問(wèn):
1.你知道設(shè)計(jì)模式嗎骚露?--當(dāng)然障般。
2.告訴我一個(gè)商業(yè)案例并寫(xiě)下工廠方法的代碼片段。--寫(xiě)下了一個(gè)簡(jiǎn)單工廠模式(非常不幸盛杰,簡(jiǎn)單工廠模式不被認(rèn)為是設(shè)計(jì)模式)。
在本文即供,我不會(huì)展示如何編寫(xiě)Simple Factory和Factory Method ,相反我更愿意討論它們是什么以及何時(shí)使用它們。
如果你關(guān)于上面的兩個(gè)問(wèn)題的答案和上面的一樣逗嫡,要注意了:
1青自、實(shí)際上是在試圖記住設(shè)計(jì)模式,而不是理解它試圖解決的問(wèn)題驱证。
2延窜、我們無(wú)法分析業(yè)務(wù)的問(wèn)題并基于它們應(yīng)用模式。相反抹锄,我們?cè)噲D在業(yè)務(wù)案例中找到我們所知道的模式逆瑞。
簡(jiǎn)單工廠模式
// 外賣(mài)接口
public interface Takeaway {
String getTakeawayName();
}
// 百度糯米 外賣(mài)
public class BaiDuNuomiTakeaway implements Takeaway{
@Override
public String getTakeawayName() {
return "百度糯米";
}
}
// 美團(tuán) 外賣(mài)
public class MeiTuanTakeaway implements Takeaway{
@Override
public String getTakeawayName() {
return "美團(tuán)";
}
}
// ** 工廠 **
public class SimpleFactory {
public static Takeaway createTakeaway(String type){
switch (type){
case "百度糯米": return new BaiDuNuomiTakeaway();
case "美團(tuán)": return new MeiTuanTakeaway();
default: return null;
}
}
public static void main(String[] args) {
System.out.println(SimpleFactory.createTakeaway("百度糯米").getTakeawayName());
}
}
使用Simple Factory 荠藤,調(diào)用者唯一知道的就是通過(guò)調(diào)用靜態(tài)方法并傳遞所需要的參數(shù),工廠就會(huì)返回一個(gè)外賣(mài)對(duì)象获高。但是如何創(chuàng)建外賣(mài)哈肖,客戶(hù)端的代碼并不知道。
改變是軟件開(kāi)發(fā)唯一不變的念秧。比如我希望工廠在創(chuàng)建餓了么淤井,當(dāng)目前的工廠不能滿足我時(shí),毫無(wú)疑問(wèn)摊趾,我需要?jiǎng)?chuàng)建餓了么的類(lèi)庄吼,并在工廠中添加餓了么的創(chuàng)建過(guò)程。在小范圍可能感覺(jué)不出來(lái)严就,但在更大的業(yè)務(wù)范圍內(nèi)总寻,這會(huì)損害設(shè)計(jì),接下來(lái)我們?cè)陂_(kāi)看看業(yè)務(wù)的簡(jiǎn)單變化梢为,會(huì)如何影響設(shè)計(jì)渐行。
工廠模式系列(工廠方法模式、抽象工廠模式)
除了Simple Factory之外,Factory Pattern系列的另外兩個(gè)成員:Factory Method和Abstract Factory铸董。我不在此詳細(xì)介紹這兩種模式了祟印。
簡(jiǎn)而言之,F(xiàn)actory Method 使用實(shí)現(xiàn)接口粟害,Abstract Factory 使用繼承蕴忆。Factory Method 是關(guān)于創(chuàng)建一種類(lèi)型的對(duì)象,而Abstract Factory是關(guān)于創(chuàng)建一系列不同類(lèi)型的對(duì)象悲幅。所有這三個(gè)都是通過(guò)設(shè)計(jì)原則封裝對(duì)象創(chuàng)建:封裝變化的東西套鹅。
如果業(yè)務(wù)需求不僅僅是產(chǎn)品創(chuàng)建,如果你想要控制產(chǎn)品創(chuàng)建步驟并希望控制每一個(gè)步驟汰具,并且步驟是自定義的卓鹿,那我們可以使用工廠方法模式,換句話說(shuō)留荔,如果要控制一些列產(chǎn)品的算法策略吟孙,可以考慮 工廠方法模式。
如果要?jiǎng)?chuàng)建一個(gè)框架聚蝶,你希望從創(chuàng)建Object到管理該Object的控件杰妓,請(qǐng)使用工廠方法模式,這與簡(jiǎn)單工廠不同碘勉,簡(jiǎn)單工廠只關(guān)注產(chǎn)品創(chuàng)建后的結(jié)果巷挥,而不關(guān)心如何創(chuàng)建并管理產(chǎn)品。
回到我們的例子【涓鳎現(xiàn)在晴叨,假設(shè)我們的業(yè)務(wù)要求不僅要?jiǎng)?chuàng)建外賣(mài),還要根據(jù)當(dāng)前外賣(mài)人員距離用戶(hù)遠(yuǎn)近是否接受訂單和發(fā)送外賣(mài)初厚。
因此孙技,我們需要?jiǎng)?chuàng)建一種算法,創(chuàng)建賣(mài)并發(fā)送它們亚情,計(jì)算運(yùn)輸成本哈雏。工廠方法模式是理想的,我們?cè)谥匦略O(shè)計(jì)一下土浸。
編碼
// 抽象工廠 :生產(chǎn)不同種類(lèi)的產(chǎn)品
public abstract class AbstractFactory {
public abstract MeiTuanTakeaway createMeiTuanTakeaway();
public abstract int shippingCharge();
public final void orderTakeaway(){
createMeiTuanTakeaway();
int chagre=shippingCharge();
System.out.println("運(yùn)費(fèi):"+chagre);
}
}
// 外賣(mài)
public class Takeaway extends AbstractFactory {
@Override
public MeiTuanTakeaway createMeiTuanTakeaway() {
return new MeiTuanTakeaway();
}
@Override
public int shippingCharge() {
return 100;
}
public static void main(String[] args) {
AbstractFactory factory = new Takeaway();
factory.orderTakeaway();
}
}
理由:使用工廠方法模式黄伊,我們不僅可以抽象產(chǎn)品創(chuàng)建派殷,還可以更進(jìn)一步,我們可以抽象生產(chǎn)外賣(mài)的工廠憋活。因此我們可以控制產(chǎn)品的創(chuàng)建和管理虱黄。上面的代碼,你會(huì)發(fā)現(xiàn)橱乱,我們使用了抽象工廠,在抽象工廠中作瞄,我們創(chuàng)建了一個(gè)模板方法危纫,來(lái)控制外賣(mài)的生命周期-創(chuàng)建外賣(mài)和運(yùn)輸成本計(jì)算乌庶,并可以使抽象工廠可以相應(yīng)的改變他們瞒大。
優(yōu)點(diǎn):創(chuàng)建一個(gè)框架搪桂,控制產(chǎn)品生命周期。
總結(jié):
簡(jiǎn)單工廠模式:根據(jù)調(diào)用者傳入的參數(shù)酗电,工廠動(dòng)態(tài)決定所要?jiǎng)?chuàng)建的產(chǎn)品内列。只關(guān)心結(jié)果。
工廠方法模式:核心類(lèi)成為一個(gè)抽象工廠角色荷荤,而實(shí)現(xiàn)類(lèi)移稳,只創(chuàng)建一種類(lèi)型對(duì)象。
抽象工廠模式:創(chuàng)建一系列不同類(lèi)型對(duì)象古毛。
推薦閱讀: