設(shè)計(jì)模式之工廠模式

客戶(hù)需求

    /**
     * 小明在北京開(kāi)了一家pizza店,生意很好,此時(shí),小強(qiáng)和小紅都想加盟他的pizza店,
     * 分別在廣東和湖南開(kāi)一家pizza店。(以后可能加盟店越來(lái)越多)
     * 原料:dough, sauce, toppings,cheese(奶酪), clam(哈蜊), 
     * veggie(素食), pepperoni(意式香腸)(以后可能還有更多)
     * 
     * 制作流程:準(zhǔn)備,烘烤,切割,打包
     * 
     * 要求:1、廣東店和湖南店的口味不同,需適合當(dāng)?shù)厝说目谖?     * 
     * 2、為保證披薩質(zhì)量,加盟店必須與北京店制作流程一致
     * 
     * 3、必須防止加盟店使用低價(jià)原料來(lái)增加利潤(rùn)
     * 
     * 請(qǐng)用代碼描述以上需求
     * 
     */

程序設(shè)計(jì)

1、PizzaStore是用來(lái)給客戶(hù)下訂單買(mǎi)pizza的,所以每個(gè)PizzaStore都會(huì)有一個(gè)orderPizza的方法,返回pizza給客戶(hù);

2、當(dāng)客戶(hù)下單后,就需要生產(chǎn)對(duì)應(yīng)的pizza,PizzaStore不需要知道如何去創(chuàng)造pizza,根據(jù)客戶(hù)的需求交給對(duì)應(yīng)的子類(lèi)去完成;

3、當(dāng)去生產(chǎn)滿(mǎn)足客戶(hù)需求的pizza時(shí),我們都會(huì)用new來(lái)獲取這個(gè)pizza的實(shí)例對(duì)象,此時(shí),我們需意識(shí)到,new pizza時(shí)是整個(gè)過(guò)程變化的部分,那么就需馬上想到我們之前學(xué)習(xí)策略模式時(shí)講過(guò)的設(shè)計(jì)原則:找出程序中可能需要變化之處,把它們獨(dú)立出來(lái),不要和那些不需要變化的代碼混在一起

4、廢話(huà)不多說(shuō),代碼實(shí)現(xiàn)

  • Pizza,若以后還需要添加更多的原料,直接增加屬性就可以了

      public abstract class Pizza
      {
          /**
           * 披薩名稱(chēng)
           */
          protected String            mPizzaName;
          /**
           * 面粉類(lèi)型
           */
          protected String            mPizzaDough;
          /**
           * 醬料類(lèi)型
           */
          protected String            mPizzaSauce;
          /**
           * 其他佐料
           */
          protected ArrayList<String> mPizzaToppings  = new ArrayList<>();
      
          public void prepare()
          {
              System.out.println("準(zhǔn)備:" + mPizzaName);
              System.out.println("攪拌面粉:" + mPizzaDough);
              System.out.println("添加醬料:" + mPizzaSauce);
              for (int i = 0; i < mPizzaToppings.size(); i++)
              {
                  System.out.println("其他佐料:" + mPizzaToppings.get(i));
              }
          }
          /**
            * 不允許子類(lèi)修改烘烤時(shí)間
            */
          public final void bake()
          {
              System.out.println("大約烘烤25分鐘");
          }
      
          public void cut()
          {
              System.out.println("將披薩切成小塊三角形狀");
          }
          /**
            * 不允許子類(lèi)修改包裝方式
            */
          public final void box()
          {
              System.out.println("包裝好");
          }
      
          public String getName()
          {
              return mPizzaName;
          }
      }
    
  • PizzaStore

      public abstract class PizzaStore
      {
      
          /**
           * 根據(jù)客戶(hù)需求預(yù)訂披薩
           * 
           * @param type
           *            披薩類(lèi)型
           * @return
           */
          public Pizza orderPizza(String type)
          {
              Pizza pizza = createPizza(type);
              pizza.prepare();
              pizza.bake();
              pizza.cut();
              pizza.box();
              return pizza;
          }
          public abstract Pizza createPizza(String type);
      }
    
  • GuangDongStylePizzaStore

      public class GuangDongStylePizzaStore extends PizzaStore
      {
          @Override
          public Pizza createPizza(String type)
          {
              Pizza pizza = null;
              // 這里可以用枚舉來(lái)表示pizza的原料內(nèi)容,防止客戶(hù)輸入錯(cuò)誤。這里就偷一下懶了
              if (type.equals("cheese"))
              {
                  pizza = new GuangDongStyleCheesePizza();
              }
              else if (type.equals("pepperoni"))
              {
                  pizza = new GuangDongStylePepperoniPizza();
              }
              else if (type.equals("clam"))
              {
                  pizza = new GuangDongStyleClamPizza();
              }
              else if (type.equals("veggie"))
              {
                  pizza = new GuangDongStyleVeggiePizza();
              }
              return pizza;
          }
      }
    
  • HuNanStylePizzaStore

      public class HuNanStylePizzaStore extends PizzaStore
      {
      
          @Override
          public Pizza createPizza(String type)
          {
              Pizza pizza = null;
              if (type.equals("cheese"))
              {
                  pizza = new HuNanStyleCheesePizza();
              }
              else if (type.equals("pepperoni"))
              {
                  pizza = new HuNanStylePepperoniPizza();
              }
              else if (type.equals("clam"))
              {
                  pizza = new HuNanStyleClamPizza();
              }
              else if (type.equals("veggie"))
              {
                  pizza = new HuNanStyleVeggiePizza();
              }
              return pizza;
          }
      }
    
  • 這里只貼出兩種CheesePizza的代碼

      /**
       * 湖南口味奶酪披薩
       * 
       * 
       */
      public class HuNanStyleCheesePizza extends Pizza
      {
      
          public HuNanStyleCheesePizza() {
              mPizzaName = "HuNan Style Deep Dish Cheese Pizza";
              mPizzaDough = "Extra Thick Crust Dough";
              mPizzaSauce = "Plum Tomato Sauce";
              mPizzaToppings.add("Shredded Mozzarella Cheese");
          }
      
          @Override
          public void cut()
          {
              System.out.println("將披薩切成小塊矩形狀");
          }
    
      }
      
      -----------------------------------------------------------
      /**
       * 廣東口味奶酪披薩
       * 
       * 
       */
      public class GuangDongStyleCheesePizza extends Pizza
      {
      
          public GuangDongStyleCheesePizza() {
              mPizzaName = "New York Style Sauce and Cheese Pizza";
              mPizzaDough = "Thin Crust Dough";
              mPizzaSauce = "Marinara Sauce";
              mPizzaToppings.add("Grated Reggiano Cheese");
          }
      }
    

測(cè)試代碼

    public class FactoryDesignPatternTest
    {
    
        public static void main(String[] args)
        {
            PizzaStore huNanPizzaStore = new HuNanStylePizzaStore();
            PizzaStore guangDongPizzaStore = new GuangDongStylePizzaStore();
            Pizza huNanPizza = huNanPizzaStore.orderPizza("cheese");
            System.out.println(huNanPizza.getName());
            System.out.println("-----------------------------------");
            Pizza guangDongPizza = guangDongPizzaStore.orderPizza("cheese");
            System.out.println(guangDongPizza.getName());
    
        }
    }

測(cè)試結(jié)果

FactoryMethodPatternTest.png

工廠方法模式

  • 定義

    定義了一個(gè)創(chuàng)建對(duì)象的抽象類(lèi),但由子類(lèi)決定要實(shí)例化的類(lèi)是哪一個(gè)。工廠方法讓類(lèi)把實(shí)例化推遲到子類(lèi)

  • Demo UML圖

FactoryMethodPatternDemoUML.png

接下來(lái),我們?yōu)槊總€(gè)區(qū)域創(chuàng)建原料工廠

public interface PizzaIngredientFactory
{
   /* 
    * 創(chuàng)建原料的方法,為了方便,直接用字符串代替, 在實(shí)際項(xiàng)目中,原料可以用類(lèi)來(lái)表示
    */
  public String createDough();

  public String createSauce();

  public String createCheese();

  public String[] createVeggies();

  public String createPepperoni();

  public String createClams();
}

不同區(qū)域有不同的原料工廠

public class HuNanPizzaIngredientFactory implements PizzaIngredientFactory
{
  @Override
  public String createDough()
  {
      return "ThinCrustDough";
  }

  @Override
  public String createSauce()
  {
      return "MarinaraSauce";
  }

  @Override
  public String createCheese()
  {
      return "ReggianoCheese";
  }

  @Override
  public String[] createVeggies()
  {
      return new String[] { "Garlic", "Onion", "Mushroom", "RedPepper" };
  }

  @Override
  public String createPepperoni()
  {
      return "SlicedPepperoni";
  }

  @Override
  public String createClams()
  {
      return "FreshClams";
  }
}

 ----------------------------------------------------------------------
 public class GuangDongPizzaIngredientFactory implements PizzaIngredientFactory
 {
    @Override
    public String createDough()
    {
        return "ThickCrustDough";
    }

    @Override
    public String createSauce()
    {
        return "PlumTomatoSauce";
    }

    @Override
    public String createCheese()
    {
        return "MozzarellaCheese";
    }

    @Override
    public String[] createVeggies()
    {
        return new String[] { "BlackOlives", "Spinach", "Eggplant" };
    }

    @Override
    public String createPepperoni()
    {
        return "SlicedPepperoni";
    }

    @Override
    public String createClams()
    {
        return "FrozenClams";
    }
}

重新修改一下pizza的代碼,添加兩種原料,并抽象準(zhǔn)備流程,讓子類(lèi)自己去準(zhǔn)備需要的原料

public abstract class Pizza
{

    protected String            mPizzaName;
    protected String            mPizzaDough;
    protected String            mPizzaSauce;
    protected ArrayList<String> mPizzaToppings  = new ArrayList<>();
       //新添加的原料
    protected String            mPizzaCheese;
    protected String            mPizzaClam;
    // public void prepare()
    // {
        // System.out.println("準(zhǔn)備:" + mPizzaName);
        // System.out.println("攪拌面粉:" + mPizzaDough);
        // System.out.println("添加醬料:" + mPizzaSauce);
        // for (int i = 0; i < mPizzaToppings.size(); i++)
        // {
            // System.out.println("其他佐料:" + mPizzaToppings.get(i));
        // }
    //}
    protected abstract void prepareIngredient();
    public final void bake()
    {
        System.out.println("大約烘烤25分鐘");
    }
    public void cut()
    {
    System.out.println("將披薩切成小塊三角形狀");
    }
    public final void box()
    {
        System.out.println("包裝好");
    }
    public void setName(String name)
    {
    mPizzaName = name;
    }
    public String getName()
    {
        return mPizzaName;
    }
}    

我們不需要設(shè)計(jì)不同的類(lèi)來(lái)處理不同口味的披薩,讓原料廠處理這種區(qū)域差異就可以了。

public class CheesePizza extends Pizza
{
    private PizzaIngredientFactory  mIngredientFactory;

    public CheesePizza(PizzaIngredientFactory factory) {
        mIngredientFactory = factory;
    }

    @Override
    protected void prepareIngredient()
    {
        System.out.println("preparing:" + mPizzaName);
        mPizzaDough = mIngredientFactory.createDough();
        mPizzaSauce = mIngredientFactory.createSauce();
        mPizzaCheese = mIngredientFactory.createCheese();
    }
}

再回我到我們的Pizza店

public class HuNanStylePizzaStore extends PizzaStore
{

    @Override
    public Pizza createPizza(String type)
    {
        Pizza pizza = null;
        // 湖南Pizza店用到了湖南原料工廠,由該原料工廠負(fù)責(zé)生產(chǎn)所有湖南口味的披薩所需的原料
        PizzaIngredientFactory ingredientFactory = new HuNanPizzaIngredientFactory();

        if (type.equals("cheese"))
        {
            // 對(duì)象組合:把工廠傳遞給每一個(gè)披薩,以便披薩從工廠中取得原料
            pizza = new CheesePizza(ingredientFactory);
            pizza.setName("New York Style Cheese Pizza");
        }
        else if (type.equals("pepperoni"))
        {
            pizza = new PepperoniPizza(ingredientFactory);
            pizza.setName("New York Style Pepperoni Pizza");
        }
        else if (type.equals("clam"))
        {
            pizza = new ClamPizza(ingredientFactory);
            pizza.setName("New York Style Clam Pizza");
        }
        else if (type.equals("veggie"))
        {
            pizza = new VeggiePizza(ingredientFactory);
            pizza.setName("New York Style Veggie Pizza");
        }
        return pizza;
    }

}

測(cè)試代碼

PizzaStore NYStore = new NYStylePizzaStore();
Pizza pizzaTwo = NYStore.orderPizza("cheese");
System.out.println(pizzaTwo.getName());

此時(shí),你有沒(méi)有發(fā)現(xiàn)這個(gè)版本的createPizza()和之前的工廠方法實(shí)現(xiàn)的有什么不同?

我們引入了新類(lèi)型的工廠,也就是所謂的抽象工廠來(lái)創(chuàng)建披薩原料家族。通過(guò)抽象工廠所提供的接口,可以創(chuàng)建產(chǎn)品的家族,利用這個(gè)接口書(shū)寫(xiě)代碼,我們的代碼將從實(shí)際工廠解耦,以便在不同上下文中實(shí)現(xiàn)各式各樣的工廠,制造出各種不同的產(chǎn)品。

抽象工廠模式

  • 定義

    提供一個(gè)接口,用于創(chuàng)建相關(guān)或依賴(lài)對(duì)象的家族,而不需要明確指定具體類(lèi)

  • Demo UML類(lèi)圖

AbstractFactoryPatternDemoUML.png

你可能注意到了,抽象工廠的每個(gè)方法實(shí)際上看起來(lái)都是工廠方法,因?yàn)槊總€(gè)方法都被聲明成抽象,而子類(lèi)的方法覆蓋這些法來(lái)創(chuàng)建某些對(duì)象。

那么,工廠方法是不是潛伏在抽象工廠里面了?

是的,抽象工廠的方法經(jīng)常以工廠方法的方式實(shí)現(xiàn)。抽象工廠的任務(wù)就是定義一個(gè)負(fù)責(zé)創(chuàng)建一組產(chǎn)品的接口,這個(gè)接口內(nèi)的每個(gè)方法都負(fù)責(zé)創(chuàng)建一個(gè)具體的產(chǎn)品,同時(shí)我們利用實(shí)現(xiàn)抽象工廠的子類(lèi)來(lái)提供這些具體的做法。所以,在抽象工廠中利用工廠方法實(shí)現(xiàn)生產(chǎn)方法是相當(dāng)自然的做法。

工廠方法模式與抽象工廠模式不同之處

FactoryMethodPatternUML.png
AbstractFactoryPatternUML.png
  • 工廠方法利用繼承的方式來(lái)創(chuàng)建對(duì)象,抽象工廠則是用的對(duì)象組合來(lái)創(chuàng)建對(duì)象
  • 工廠方法只能生產(chǎn)同一等級(jí)結(jié)構(gòu)中的固定產(chǎn)品,而抽象工廠能夠生產(chǎn)不同產(chǎn)品族的全部產(chǎn)品
  • 工廠方法的優(yōu)勢(shì):支持增加任意產(chǎn)品;抽象工廠的優(yōu)勢(shì):支持增加產(chǎn)品族,對(duì)于增加新的產(chǎn)品,需改變接口,可能會(huì)造成繁重的工作;

工廠模式中用到的設(shè)計(jì)原則

依賴(lài)倒置原則(Dependency Inversion Principle)

要依賴(lài)抽象,不要依賴(lài)具體類(lèi)

這個(gè)原則似乎聽(tīng)起來(lái)很像是“針對(duì)接口編程,不針對(duì)實(shí)現(xiàn)編程”,的確很相似,但這里更強(qiáng)調(diào)“抽象”。這個(gè)原則說(shuō)明了:不能讓高層組件依賴(lài)低層組件,并且,不管高層或低層組件,“兩者”都應(yīng)該依賴(lài)于抽象。
在我們的Demo中,PizzaStore是“高層組件”,而具體的pizza是”低層組件“,他們都依賴(lài)Pizza這個(gè)抽象類(lèi)。

參考資料

Head First 設(shè)計(jì)模式

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,837評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,196評(píng)論 3 414
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 175,688評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,654評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,456評(píng)論 6 406
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 54,955評(píng)論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,044評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,195評(píng)論 0 287
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,725評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,608評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,802評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,318評(píng)論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,048評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,422評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,673評(píng)論 1 281
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,424評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,762評(píng)論 2 372

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