在簡單工廠模式中講到簡單工廠模式的缺點,有一點是——簡單工廠模式系統(tǒng)難以擴展,一旦添加新產品就不得不修改簡單工廠方法,這樣就會造成簡單工廠的實現(xiàn)邏輯過于復雜,然而本專題介紹的工廠方法模式可以解決簡單工廠模式中存在的這個問題,下面就具體看看工廠模式是如何解決該問題的。
網站建設哪家好,找創(chuàng)新互聯(lián)!專注于網頁設計、網站建設、微信開發(fā)、微信小程序開發(fā)、集團企業(yè)網站建設等服務項目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了遷安免費建站歡迎大家使用!
工廠方法模式之所以可以解決簡單工廠的模式,是因為它的實現(xiàn)把具體產品的創(chuàng)建推遲到子類中,此時工廠類不再負責所有產品的創(chuàng)建,而只是給出具體工廠必須實現(xiàn)的接口,這樣工廠方法模式就可以允許系統(tǒng)不修改工廠類邏輯的情況下來添加新產品,這樣也就克服了簡單工廠模式中缺點。下面看下工廠模式的具體實現(xiàn)代碼(這里還是以簡單工廠模式中點菜的例子來實現(xiàn)):
namespace 設計模式之工廠方法模式 { ////// 菜抽象類 /// public abstract class Food { // 輸出點了什么菜 public abstract void Print(); } ////// 西紅柿炒雞蛋這道菜 /// public class TomatoScrambledEggs : Food { public override void Print() { Console.WriteLine("西紅柿炒蛋好了!"); } } ////// 土豆肉絲這道菜 /// public class ShreddedPorkWithPotatoes : Food { public override void Print() { Console.WriteLine("土豆肉絲好了"); } } ////// 抽象工廠類 /// public abstract class Creator { // 工廠方法 public abstract Food CreateFoddFactory(); } ////// 西紅柿炒蛋工廠類 /// public class TomatoScrambledEggsFactory:Creator { ////// 負責創(chuàng)建西紅柿炒蛋這道菜 /// ///public override Food CreateFoddFactory() { return new TomatoScrambledEggs(); } } /// /// 土豆肉絲工廠類 /// public class ShreddedPorkWithPotatoesFactory:Creator { ////// 負責創(chuàng)建土豆肉絲這道菜 /// ///public override Food CreateFoddFactory() { return new ShreddedPorkWithPotatoes(); } } /// /// 客戶端調用 /// class Client { static void Main(string[] args) { // 初始化做菜的兩個工廠() Creator shreddedPorkWithPotatoesFactory = new ShreddedPorkWithPotatoesFactory(); Creator tomatoScrambledEggsFactory = new TomatoScrambledEggsFactory(); // 開始做西紅柿炒蛋 Food tomatoScrambleEggs = tomatoScrambledEggsFactory.CreateFoddFactory(); tomatoScrambleEggs.Print(); //開始做土豆肉絲 Food shreddedPorkWithPotatoes = shreddedPorkWithPotatoesFactory.CreateFoddFactory(); shreddedPorkWithPotatoes.Print(); Console.Read(); } } }
使用工廠方法實現(xiàn)的系統(tǒng),如果系統(tǒng)需要添加新產品時,我們可以利用多態(tài)性來完成系統(tǒng)的擴展,對于抽象工廠類和具體工廠中的代碼都不需要做任何改動。例如,我們我們還想點一個“肉末茄子”,此時我們只需要定義一個肉末茄子具體工廠類和肉末茄子類就可以。而不用像簡單工廠模式中那樣去修改工廠類中的實現(xiàn)(具體指添加case語句)。具體代碼為:
////// 肉末茄子這道菜 /// public class MincedMeatEggplant : Food { ////// 重寫抽象類中的方法 /// public override void Print() { Console.WriteLine("肉末茄子好了"); } } ////// 肉末茄子工廠類,負責創(chuàng)建肉末茄子這道菜 /// public class MincedMeatEggplantFactory : Creator { ////// 負責創(chuàng)建肉末茄子這道菜 /// ///public override Food CreateFoddFactory() { return new MincedMeatEggplant(); } } /// /// 客戶端調用 /// class Client { static void Main(string[] args) { // 如果客戶又想點肉末茄子了 // 再另外初始化一個肉末茄子工廠 Creator minceMeatEggplantFactor = new MincedMeatEggplantFactory(); // 利用肉末茄子工廠來創(chuàng)建肉末茄子這道菜 Food minceMeatEggplant = minceMeatEggplantFactor.CreateFoddFactory(); minceMeatEggplant.Print(); Console.Read(); } }
講解完工廠模式的具體實現(xiàn)之后,讓我們看下工廠模式中各類之間的UML圖:
從UML圖可以看出,在工廠方法模式中,工廠類與具體產品類具有平行的等級結構,它們之間是一一對應的。針對UML圖的解釋如下:
Creator類:充當抽象工廠角色,任何具體工廠都必須繼承該抽象類
TomatoScrambledEggsFactory和ShreddedPorkWithPotatoesFactory類:充當具體工廠角色,用來創(chuàng)建具體產品
Food類:充當抽象產品角色,具體產品的抽象類。任何具體產品都應該繼承該類
TomatoScrambledEggs和ShreddedPorkWithPotatoes類:充當具體產品角色,實現(xiàn)抽象產品類對定義的抽象方法,由具體工廠類創(chuàng)建,它們之間有一一對應的關系。
.NET 類庫中也有很多實現(xiàn)了工廠方法的類,例如Asp.net中,處理程序對象是具體用來處理請求,當我們請求一個*.aspx的文件時,此時會映射到System.Web.UI.PageHandlerFactory類上進行處理,而對*.ashx的請求將映射到System.Web.UI.SimpleHandlerFactory類中(這兩個類都是繼承于IHttpHandlerFactory接口的),關于這點說明我們可以在“C:\Windows\Microsoft.NET\Framework\v4.0.30319\Config\Web.Config”文件中找到相關定義,具體定義如下:
下面我們就具體看下工廠方法模式在Asp.net中是如何實現(xiàn)的,如果對一個Index.aspx頁面發(fā)出請求時,將會調用PageHandlerFactory中GetHandler方法來創(chuàng)建一個Index.aspx對象,它們之間的類圖關系如下:
工廠方法模式通過面向對象編程中的多態(tài)性來將對象的創(chuàng)建延遲到具體工廠中,從而解決了簡單工廠模式中存在的問題,也很好地符合了開放封閉原則(即對擴展開發(fā),對修改封閉)。