工廠模式

需求

開披薩店,披薩有很多種豁跑,披薩店也可以有很多個(gè)分店

目的

對(duì)修改關(guān)閉廉涕,針對(duì)接口編程,不針對(duì)實(shí)現(xiàn)編程

簡(jiǎn)單工廠

披薩

public abstract class Pizza
{
    String name;
    String dough;// 面團(tuán)
    String sauce;// 醬
    ArrayList<String> toppings = new ArrayList<String>();// 澆在披薩上的

    public String getName()
    {
        return name;
    }

    // 準(zhǔn)備
    public void prepare()
    {
        System.out.println("Prepare " + name);
        System.out.println("Tossing dough...");
        System.out.println("Adding sauce...");
        System.out.println("Adding toppings: ");
        for (String topping : toppings)
        {
            System.out.println("   " + topping);
        }
    }

    // 烘烤
    public void bake()
    {
        System.out.println("Baking " + name);
    }

    // 切片
    public void cut()
    {
        System.out.println("Cutting " + name);
    }

    // 打包
    public void box()
    {
        System.out.println("Boxing " + name);
    }

    public String toString()
    {
        // 顯示披薩名稱和配料
        StringBuffer display = new StringBuffer();
        display.append("---- " + name + " ----\n");
        display.append(dough + "\n");
        display.append(sauce + "\n");
        for (String topping : toppings)
        {
            display.append(topping + "\n");
        }
        return display.toString();
    }
}

// ------------------------------------------------------------------------

// 蛤蜊披薩
public class ClamPizza extends Pizza
{
    public ClamPizza()
    {
        name = "Clam Pizza";
        dough = "Thin crust";
        sauce = "White garlic sauce";
        toppings.add("Clams");
        toppings.add("Grated parmesan cheese");
    }
}

<br />


<br />

披薩店

public class PizzaStore
{
    SimplePizzaFactory factory; // 披薩店接收一個(gè)披薩制造工廠

    public PizzaStore(SimplePizzaFactory factory)
    {
        this.factory = factory;
    }

    // 客戶可以訂披薩
    public Pizza orderPizza(String type)
    {
        Pizza pizza;

        pizza = factory.createPizza(type); // 交給工廠來創(chuàng)建披薩

        pizza.prepare();
        pizza.bake();
        pizza.cut();
        pizza.box();

        return pizza;
    }
}

<br />


<br />

披薩工廠

public class SimplePizzaFactory
{

    public Pizza createPizza(String type)
    {
        Pizza pizza = null;

        if (type.equals("cheese"))
        {
            pizza = new CheesePizza();
        } else if (type.equals("pepperoni"))
        {
            pizza = new PepperoniPizza();
        } else if (type.equals("clam"))
        {
            pizza = new ClamPizza();
        } else if (type.equals("veggie"))
        {
            pizza = new VeggiePizza();
        }
        return pizza;
    }
}

<br />


<br />

測(cè)試

public static void main(String[] args)
{
    // 創(chuàng)建個(gè)工廠
    SimplePizzaFactory factory = new SimplePizzaFactory();
    // 創(chuàng)建個(gè)披薩店
    PizzaStore store = new PizzaStore(factory);

    // 預(yù)訂披薩艇拍,披薩店使用工廠創(chuàng)建披薩狐蜕,然后披薩店自己處理披薩的烘烤、切片卸夕、裝盒
    Pizza pizza = store.orderPizza("cheese");
    System.out.println("We ordered a " + pizza.getName() + "\n");
    System.out.println(pizza);

    pizza = store.orderPizza("veggie");
    System.out.println("We ordered a " + pizza.getName() + "\n");
    System.out.println(pizza);
}

簡(jiǎn)單工廠的問題和好處

  • 把創(chuàng)建披薩的代碼放到了一個(gè)地方层释,SimplePizzaFactory的客戶可以有很多,不只是orderPizza快集,可以是菜單什么的贡羔,通過工廠獲取價(jià)格
  • 可以使用靜態(tài)方法來替代工廠,成為靜態(tài)工廠个初,但是缺點(diǎn)是沒法通過繼承來改變行為

<br />


<br />

工廠方法乖寒,重做披薩店

由于很多加盟商,在不同的地區(qū)開店院溺,因此需要配合當(dāng)?shù)氐目谖堕灌遥孕枰ㄖ婆_店

披薩店重做

public abstract class PizzaStore
{
    // 這個(gè)方法就像一個(gè)工廠,由子類決定如何制造披薩
    public abstract Pizza createPizza(String item);

    public Pizza orderPizza(String type)
    {
        // 創(chuàng)建披薩珍逸,由子類來決定逐虚,你要?jiǎng)?chuàng)建的是什么樣的披薩
        Pizza pizza = createPizza(type);
        System.out.println("--- Making a " + pizza.getName() + " ---");
        // 加工,由于披薩的風(fēng)味和制作方法各有不同弄息,因此也交給子類來決定痊班,從而達(dá)到解耦的目的
        pizza.prepare();
        pizza.bake();
        pizza.cut();
        pizza.box();
        return pizza;
    }
}

// 開店
public class ChicagoPizzaStore extends PizzaStore
{

    public Pizza createPizza(String item)
    {
        if (item.equals("cheese"))
        {
            return new ChicagoStyleCheesePizza();
        } else if (item.equals("veggie"))
        {
            return new ChicagoStyleVeggiePizza();
        } else if (item.equals("clam"))
        {
            return new ChicagoStyleClamPizza();
        } else if (item.equals("pepperoni"))
        {
            return new ChicagoStylePepperoniPizza();
        } else
            return null;
    }
}

披薩可以自己定義加工自己的方式

public class ChicagoStyleClamPizza extends Pizza
{
    public ChicagoStyleClamPizza()
    {
        name = "Chicago Style Clam Pizza";
        dough = "Extra Thick Crust Dough";
        sauce = "Plum Tomato Sauce";

        toppings.add("Shredded Mozzarella Cheese");
        toppings.add("Frozen Clams from Chesapeake Bay");
    }

    public void cut()
    {
        System.out.println("Cutting the pizza into square slices");
    }
}

工廠方法模式的定義

定義了一個(gè)創(chuàng)建對(duì)象的接口,由子類決定要?jiǎng)?chuàng)建的對(duì)象是哪一個(gè)摹量,工廠方法讓類實(shí)例化推遲到了子類中涤伐。

設(shè)計(jì)原則 6 :依賴倒置原則

要依賴抽象馒胆,不要依賴具體類。

不能讓高層組件依賴于具體的低層組件凝果,在這個(gè)例子中祝迂,披薩店就是高層,披薩就是低層器净,而不管對(duì)于高層還是低層型雳,都應(yīng)該依賴于抽象,而不要依賴于具體的抽象類山害。

比如上面的PizzaStore依賴的就是Pizza這個(gè)抽象纠俭,而不是依賴于具體的某個(gè)Pizza子類

如何應(yīng)用此原則

  • 變量不可以持有具體類的引用,比如使用new得到具體類的實(shí)例浪慌,可以使用工廠方法
  • 類不可以派生于具體類冤荆,需要派生于一個(gè)抽象,就是接口或抽象類
  • 不要覆蓋父類中已實(shí)現(xiàn)的方法权纤,父類中的方法應(yīng)該被所有的子類共享

<br />


<br />

抽象工廠

現(xiàn)在不僅披薩店要定制钓简,制作披薩的原料也要定制会宪,因此需要不同的原料工廠

原料工廠

// 原料工廠伴澄,每種原料都對(duì)應(yīng)一種創(chuàng)建的方法,抽象化奕短,交給子類決定使用什么原料
public interface PizzaIngredientFactory
{
    public Dough createDough();

    public Sauce createSauce();

    public Cheese createCheese();

    public Veggies[] createVeggies();

    public Pepperoni createPepperoni();

    public Clams createClam();
}


// 芝加哥原料工廠古掏,指定要使用哪些原料
public class ChicagoPizzaIngredientFactory implements PizzaIngredientFactory
{
    public Dough createDough()
    {
        return new ThickCrustDough();
    }

    public Sauce createSauce()
    {
        return new PlumTomatoSauce();
    }

    public Cheese createCheese()
    {
        return new MozzarellaCheese();
    }

    public Veggies[] createVeggies()
    {
        Veggies veggies[] =
        { new BlackOlives(), new Spinach(), new Eggplant() };
        return veggies;
    }

    public Pepperoni createPepperoni()
    {
        return new SlicedPepperoni();
    }

    public Clams createClam()
    {
        return new FrozenClams();
    }
}

原料接口损话,同一種原料的生產(chǎn)之類的可能每個(gè)地區(qū)都不一樣

public interface Cheese
{
    public String toString();
}

public class MozzarellaCheese implements Cheese
{
    public String toString()
    {
        return "Shredded Mozzarella";
    }
}

// 省略其他原料定義

披薩

public abstract class Pizza
{
    public String name;

    public Dough dough;
    public Sauce sauce;
    public Veggies veggies[];
    public Cheese cheese;
    public Pepperoni pepperoni;
    public Clams clam;

    public abstract void prepare();

    public void bake()
    {
        System.out.println("Bake for 25 minutes at 350");
    }

    public void cut()
    {
        System.out.println("Cutting the pizza into diagonal slices");
    }

    public void box()
    {
        System.out.println("Place pizza in official PizzaStore box");
    }

    public void setName(String name)
    {
        this.name = name;
    }

    public String getName()
    {
        return name;
    }

    public String toString()
    {
        StringBuffer result = new StringBuffer();
        result.append("---- " + name + " ----\n");
        if (dough != null)
        {
            result.append(dough);
            result.append("\n");
        }
        if (sauce != null)
        {
            result.append(sauce);
            result.append("\n");
        }
        if (cheese != null)
        {
            result.append(cheese);
            result.append("\n");
        }
        if (veggies != null)
        {
            for (int i = 0; i < veggies.length; i++)
            {
                result.append(veggies[i]);
                if (i < veggies.length - 1)
                {
                    result.append(", ");
                }
            }
            result.append("\n");
        }
        if (clam != null)
        {
            result.append(clam);
            result.append("\n");
        }
        if (pepperoni != null)
        {
            result.append(pepperoni);
            result.append("\n");
        }
        return result.toString();
    }
}


public class CheesePizza extends Pizza
{
    PizzaIngredientFactory ingredientFactory;

    // 由商店決定要用哪個(gè)工廠的原料
    public CheesePizza(PizzaIngredientFactory ingredientFactory)
    {
        this.ingredientFactory = ingredientFactory;
    }

    public void prepare()
    {
        System.out.println("Preparing " + name);
        // 只要是工廠就可以,不在乎具體是誰(shuí)
        dough = ingredientFactory.createDough();
        sauce = ingredientFactory.createSauce();
        cheese = ingredientFactory.createCheese();
    }
}

披薩店

父類不需要更改了冗茸,由子類自己決定自己的店需要的原料工廠

public class ChicagoPizzaStore extends PizzaStore
{

    protected Pizza createPizza(String item)
    {
        Pizza pizza = null;
        // 指定工廠
        PizzaIngredientFactory ingredientFactory = new ChicagoPizzaIngredientFactory();

        if (item.equals("cheese"))
        {
            pizza = new CheesePizza(ingredientFactory);
            pizza.setName("Chicago Style Cheese Pizza");

        } else if (item.equals("veggie"))
        {
            pizza = new VeggiePizza(ingredientFactory);
            pizza.setName("Chicago Style Veggie Pizza");

        } else if (item.equals("clam"))
        {
            pizza = new ClamPizza(ingredientFactory);
            pizza.setName("Chicago Style Clam Pizza");

        } else if (item.equals("pepperoni"))
        {
            pizza = new PepperoniPizza(ingredientFactory);
            pizza.setName("Chicago Style Pepperoni Pizza");

        }
        return pizza;
    }
}

抽象工廠的定義

提供一個(gè)接口席镀,用于創(chuàng)建相關(guān)的,或者依賴的家族夏漱,而不需要明確指定具體類

工廠方法和抽象工廠的比較

  • 抽象工廠的方法,經(jīng)常以工廠方法的方式呈現(xiàn)

  • 工廠方法使用的是繼承顶捷,繼承一個(gè)工廠方法去創(chuàng)建對(duì)象挂绰,客戶只需要知道它的父抽象類型即可。抽象工廠使用的是組合服赎,在繼承到方法創(chuàng)建對(duì)象后葵蒂,然后將相關(guān)的產(chǎn)品組合起來.

  • 抽象工廠由于是大的產(chǎn)品家族,后續(xù)要是加入新的類型重虑,就需要更改接口践付,工作量會(huì)很大

  • 總之,需要產(chǎn)品族的用抽象工廠缺厉,如果只是單一的永高,可以用工廠方法

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末隧土,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子命爬,更是在濱河造成了極大的恐慌曹傀,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,406評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件饲宛,死亡現(xiàn)場(chǎng)離奇詭異皆愉,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)艇抠,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門幕庐,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人家淤,你說我怎么就攤上這事翔脱。” “怎么了媒鼓?”我有些...
    開封第一講書人閱讀 163,711評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵届吁,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我绿鸣,道長(zhǎng)疚沐,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,380評(píng)論 1 293
  • 正文 為了忘掉前任潮模,我火速辦了婚禮亮蛔,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘擎厢。我一直安慰自己究流,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,432評(píng)論 6 392
  • 文/花漫 我一把揭開白布动遭。 她就那樣靜靜地躺著芬探,像睡著了一般。 火紅的嫁衣襯著肌膚如雪厘惦。 梳的紋絲不亂的頭發(fā)上偷仿,一...
    開封第一講書人閱讀 51,301評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音宵蕉,去河邊找鬼酝静。 笑死,一個(gè)胖子當(dāng)著我的面吹牛羡玛,可吹牛的內(nèi)容都是我干的别智。 我是一名探鬼主播,決...
    沈念sama閱讀 40,145評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼稼稿,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼薄榛!你這毒婦竟也來了讳窟?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,008評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤蛇数,失蹤者是張志新(化名)和其女友劉穎挪钓,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體耳舅,經(jīng)...
    沈念sama閱讀 45,443評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡碌上,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,649評(píng)論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了浦徊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片馏予。...
    茶點(diǎn)故事閱讀 39,795評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖盔性,靈堂內(nèi)的尸體忽然破棺而出霞丧,到底是詐尸還是另有隱情,我是刑警寧澤冕香,帶...
    沈念sama閱讀 35,501評(píng)論 5 345
  • 正文 年R本政府宣布蛹尝,位于F島的核電站,受9級(jí)特大地震影響悉尾,放射性物質(zhì)發(fā)生泄漏突那。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,119評(píng)論 3 328
  • 文/蒙蒙 一构眯、第九天 我趴在偏房一處隱蔽的房頂上張望愕难。 院中可真熱鬧,春花似錦惫霸、人聲如沸猫缭。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)猜丹。三九已至,卻和暖如春茫打,著一層夾襖步出監(jiān)牢的瞬間居触,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工老赤, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人制市。 一個(gè)月前我還...
    沈念sama閱讀 47,899評(píng)論 2 370
  • 正文 我出身青樓抬旺,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親祥楣。 傳聞我的和親對(duì)象是個(gè)殘疾皇子开财,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,724評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 設(shè)計(jì)原則: 要依賴抽象汉柒,不要依賴具體類 目錄 本文的結(jié)構(gòu)如下: 什么是抽象工廠模式 為什么要用該模式 模式的結(jié)構(gòu) ...
    w1992wishes閱讀 1,114評(píng)論 0 6
  • 工廠模式包含三種模式:簡(jiǎn)單工廠模式、工廠方法模式和抽象工廠模式责鳍。 簡(jiǎn)單工廠模式 定義簡(jiǎn)單工廠模式:由一個(gè)工廠類根據(jù)...
    RickGe閱讀 354評(píng)論 0 0
  • 今天學(xué)習(xí)下最常見的工廠模式碾褂,工廠模式細(xì)分下來有三大類: 他們的目標(biāo)都是一樣的:封裝對(duì)象的創(chuàng)建。但是實(shí)現(xiàn)手段和使用場(chǎng)...
    西木柚子閱讀 2,151評(píng)論 7 28
  • 該文章屬于劉小壯原創(chuàng)历葛,轉(zhuǎn)載請(qǐng)注明:劉小壯[http://www.reibang.com/u/2de707c93d...
    劉小壯閱讀 12,761評(píng)論 29 59
  • 客戶需求 程序設(shè)計(jì) 1正塌、PizzaStore是用來給客戶下訂單買pizza的,所以每個(gè)PizzaStore都會(huì)有一...
    BlainPeng閱讀 554評(píng)論 0 2