.Net創(chuàng)建型設(shè)計(jì)模式之工廠方法模式(Factory?Method)
一、動機(jī)(Motivation)
在軟件系統(tǒng)創(chuàng)建過程中,經(jīng)常面臨著“某個對象”的創(chuàng)建工作:由于需求的變化,這個對象(的具體實(shí)現(xiàn))經(jīng)常面臨著劇烈的變化,但是它卻擁有比較穩(wěn)定的接口。
如何應(yīng)對這種變化?如何提供一種“封裝機(jī)制”來隔離出“這個易變對象”的變化,從而保持系統(tǒng)中“其他依賴對象的對象”不隨著需求改變而改變?
二、意圖(Intent)
定義一個用于創(chuàng)建對象的接口,讓子類決定實(shí)例化哪一個類。Factory Method使得一個類的實(shí)例化延遲到子類。——《設(shè)計(jì)模式》GoF
三、結(jié)構(gòu)(Structure)

四、模式的組成
可以看出,在工廠方法模式的結(jié)構(gòu)圖有以下角色:
(1)、抽象工廠角色(Creator): 充當(dāng)抽象工廠角色,定義工廠類所具有的基本的操作,任何具體工廠都必須繼承該抽象類。
(2)、具體工廠角色(ConcreteCreator):充當(dāng)具體工廠角色,該類必須繼承抽象工廠角色,實(shí)現(xiàn)抽象工廠定義的方法,用來創(chuàng)建具體產(chǎn)品。
(3)、抽象產(chǎn)品角色(Product):充當(dāng)抽象產(chǎn)品角色,定義了產(chǎn)品類型所有具有的基本操作,具體產(chǎn)品必須繼承該抽象類。
(4)、具體產(chǎn)品角色(ConcreteProduct):充當(dāng)具體產(chǎn)品角色,實(shí)現(xiàn)抽象產(chǎn)品類對定義的抽象方法,由具體工廠類創(chuàng)建,它們之間有一一對應(yīng)的關(guān)系。
五、工廠方法模式的代碼實(shí)現(xiàn)
【簡單工廠模式】的問題是:如果有新的需求就需要修改工廠類里面創(chuàng)建產(chǎn)品對象實(shí)例的那個方法的實(shí)現(xiàn)代碼,在面向?qū)ο笤O(shè)計(jì)一個原則就是哪里有變化,我就封裝哪里。
為了應(yīng)對改變,我們需要把Car先變成抽象類。
public abstract class Car
{
public abstract void startup();
public abstract void run();
public abstract void stop();
}
internal class BenzCar : Car
{
public override void startup()
{
Console.WriteLine("BenzCar Startup!");
}
public override void run()
{
Console.WriteLine("BenzCar Running!");
}
public override void stop()
{
Console.WriteLine("BenzCar Stopped!");
}
}
internal class HondaCar : Car
{
public override void startup()
{
Console.WriteLine("HondaCar Startup!");
}
public override void run()
{
Console.WriteLine("HondaCar Running!");
}
public override void stop()
{
Console.WriteLine("HondaCar Stopped!");
}
}
public abstract class CarFactory
{
public abstract Car CreatCar();
}
internal class BenzCarFactory : CarFactory
{
public override Car CreatCar()
{
return new BenzCar();
}
}
internal class HondaCarFactory : CarFactory
{
public override Car CreatCar()
{
return new HondaCar();
}
}我們在客戶程序使用的時候,把所有的Car都換成抽象的AbstractCar,這樣客戶程序就不需要了解具體測試的是哪個Car了??蛻舫绦蛉缦拢?/p>
internal class CarTestFrameWork
{
public void DoTest(CarFactory carFactory)
{
Car car = carFactory.CreatCar();
car.startup();
car.run();
car.stop();
}
}在應(yīng)用程序調(diào)用的時候,傳入客戶程序的工廠應(yīng)該是具體的HongqiCarFactory工廠。當(dāng)想換具體Car的時候,只需要創(chuàng)建一個新的Car繼承自AbstractCar,并新建一個具體CarFactory工廠繼承自抽象CarFactory。然后在具體的應(yīng)用中把具體的Car工廠參數(shù)修改即可。當(dāng)然,完全可以讓具體應(yīng)用的代碼也不用修改,把變化轉(zhuǎn)嫁到配置文件中去。
internal class Test
{
public static void Main()
{
CarTestFrameWork cf = new CarTestFrameWork();
cf.DoTest(new BenzCarFactory());
cf.DoTest(new HondaCarFactory());
}
}六、Factory Method模式的幾個要點(diǎn)
Factory Method模式主要用于隔離類對象的使用者和具體類型之間的耦合關(guān)系。面對一個經(jīng)常變化的具體類型,緊耦合關(guān)系會導(dǎo)致軟件的脆弱。
Factory Method模式通過面向?qū)ο蟮氖址?,將所要?chuàng)建的具體對象工作延遲到子類,從而實(shí)現(xiàn)一種擴(kuò)展(而非更改)的策略,較好地解決了這種緊耦合關(guān)系。
- Factory Method模式解決“單個對象”的需求變化;
- AbstractFactory模式解決“系列對象”的需求變化;
- Builder模式解決“對象部分”的需求變化;
1、工廠方法模式的優(yōu)點(diǎn):
(1)、 在工廠方法中,用戶只需要知道所要產(chǎn)品的具體工廠,無須關(guān)系具體的創(chuàng)建過程,甚至不需要具體產(chǎn)品類的類名。
(2)、在系統(tǒng)增加新的產(chǎn)品時,我們只需要添加一個具體產(chǎn)品類和對應(yīng)的實(shí)現(xiàn)工廠,無需對原工廠進(jìn)行任何修改,很好地符合了“開閉原則”。
2、工廠方法模式的缺點(diǎn):
(1)、每次增加一個產(chǎn)品時,都需要增加一個具體類和對象實(shí)現(xiàn)工廠,是的系統(tǒng)中類的個數(shù)成倍增加,在一定程度上增加了系統(tǒng)的復(fù)雜度,同時也增加了系統(tǒng)具體類的依賴。這并不是什么好事。
3、工廠方法模式使用的場景:
(1)、一個類不知道它所需要的對象的類。在工廠方法模式中,我們不需要具體產(chǎn)品的類名,我們只需要知道創(chuàng)建它的具體工廠即可。
(2)、一個類通過其子類來指定創(chuàng)建那個對象。在工廠方法模式中,對于抽象工廠類只需要提供一個創(chuàng)建產(chǎn)品的接口,而由其子類來確定具體要創(chuàng)建的對象,在程序運(yùn)行時,子類對象將覆蓋父類對象,從而使得系統(tǒng)更容易擴(kuò)展。
(3)、將創(chuàng)建對象的任務(wù)委托給多個工廠子類中的某一個,客戶端在使用時可以無須關(guān)心是哪一個工廠子類創(chuàng)建產(chǎn)品子類,需要時再動態(tài)指定。
七、.NET中實(shí)現(xiàn)了工廠方法的類
.NET 類庫中也有很多實(shí)現(xiàn)了工廠方法的類,例如Asp.net中,處理程序?qū)ο笫蔷唧w用來處理請求,當(dāng)我們請求一個*.aspx的文件時,此時會映射到System.Web.UI.PageHandlerFactory類上進(jìn)行處理,而對*.ashx的請求將映射到System.Web.UI.SimpleHandlerFactory類中(這兩個類都是繼承于IHttpHandlerFactory接口的),關(guān)于這點(diǎn)說明我們可以在“C:\Windows\Microsoft.NET\Framework\v4.0.30319\Config\Web.Config”文件中找到相關(guān)定義,具體定義如下:
<add path="*.axd" verb="*" type="System.Web.HttpNotFoundHandler" validate="True"/> <add path="*.aspx" verb="*" type="System.Web.UI.PageHandlerFactory" validate="True"/> <add path="*.ashx" verb="*" type="System.Web.UI.SimpleHandlerFactory" validate="True"/>
配置文件截圖了一部分,有時間大家可以自己去研究一下。
下面我們就具體看下工廠方法模式在Asp.net中是如何實(shí)現(xiàn)的,如果對一個Index.aspx頁面發(fā)出請求時,將會調(diào)用PageHandlerFactory中GetHandler方法來創(chuàng)建一個Index.aspx對象,它們之間的類圖關(guān)系如下:

到此這篇關(guān)于.Net設(shè)計(jì)模式之工廠方法模式(Factory Method)的文章就介紹到這了。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
.Net?Core中使用MongoDB搭建集群與項(xiàng)目實(shí)戰(zhàn)
本文詳細(xì)講解了.Net?Core中使用MongoDB搭建集群與項(xiàng)目實(shí)戰(zhàn),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-02-02
ASP.NET 5已終結(jié),迎來ASP.NET Core 1.0和.NET Core 1.0
命名是非常困難的事情,微軟這次為了和ASP.NET4.6做區(qū)分,采用了全新的命名方式ASP.NET Core 1.0,它是一個全新的框架。2016-03-03
.Net結(jié)構(gòu)型設(shè)計(jì)模式之橋接模式(Bridge)
這篇文章介紹了.Net結(jié)構(gòu)型設(shè)計(jì)模式之橋接模式(Bridge),文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-05-05
讓Sqlite脫離VC++ Runtime獨(dú)立運(yùn)行的方法
這篇文章主要介紹了讓Sqlite脫離VC++ Runtime獨(dú)立運(yùn)行的方法,需要的朋友可以參考下2014-07-07

