电竞比分网-中国电竞赛事及体育赛事平台

分享

設(shè)計(jì)模式學(xué)習(xí)筆記(五)——FactoryMethod工廠模式 - KiddLee - 博客園

 skywood 2007-08-06

       剛開(kāi)始接觸設(shè)計(jì)模式時(shí)就常常聽(tīng)到同事提起工廠模式,那時(shí)也看過(guò)一些人寫(xiě)的Blog, 但是往往把注意力放在代碼的編寫(xiě)上。在這段時(shí)間的學(xué)習(xí)中慢慢體會(huì)到設(shè)計(jì)模式是用來(lái)解決一類(lèi)問(wèn)題,而不是某些固定的代碼片段。換句話說(shuō)是解決問(wèn)題的思想。設(shè) 計(jì)模式可以解決模塊的耦合關(guān)系,可以解決因需求變動(dòng)帶來(lái)的問(wèn)題。程序在第一次編寫(xiě)時(shí),各個(gè)模塊之間可能是緊耦合,但是經(jīng)過(guò)代碼重構(gòu),可以將模塊之間變?yōu)樗? 耦合。當(dāng)然,我覺(jué)得我們也可以在軟件設(shè)計(jì)之初把變化考慮到其中,對(duì)于業(yè)務(wù)型軟件設(shè)計(jì),在了解需求后,可以盡可能將其分出主次關(guān)系。也就是主體與枝節(jié)的關(guān) 系。如下圖


      
對(duì) 于工廠模式來(lái)說(shuō),要求高層模塊變化相對(duì)較慢,底層模塊變化相對(duì)較快。這樣符合設(shè)計(jì)模式中的依賴倒置原則——高層模塊不依賴于底層模塊。換句話說(shuō),軟件設(shè)計(jì) 是要?jiǎng)澐忠鬃儾糠趾头€(wěn)定部分。這樣在一些枝節(jié)問(wèn)題發(fā)生變化時(shí),主干不用變化。如果是緊耦合狀態(tài),有可能造成一個(gè)地方變化,連帶著很多地方要發(fā)生變化。

       工廠模式要解決的是“某個(gè)對(duì)象”的創(chuàng)建工作,由于需求的變化,這個(gè)對(duì)象常常面臨著劇烈的變化,但是這個(gè)對(duì)象擁有的接口相對(duì)穩(wěn)定。也就是說(shuō):枝節(jié)常常發(fā)生變化,但是枝節(jié)與主干的接口相對(duì)穩(wěn)定。

       《設(shè)計(jì)模式》中是這樣說(shuō)明:定義一個(gè)用于創(chuàng)建對(duì)象的接口,讓子類(lèi)決定實(shí)例化那個(gè)類(lèi)。FactoryMethod使得一個(gè)類(lèi)的實(shí)例化延遲到子類(lèi)。

       現(xiàn)在看這句話可能有些不明白,我們一會(huì)再來(lái)分析一下。先來(lái)看看工廠模式的大體結(jié)構(gòu)。如下圖:


      
我們還是用實(shí)例化汽車(chē)的例子來(lái)解釋。對(duì)于客戶端程序(ClientApp)如果想要一個(gè)汽車(chē)的對(duì)象,需要調(diào)用生產(chǎn)這個(gè)汽車(chē)的Factory的對(duì)象。當(dāng)然,這個(gè)類(lèi)繼承自一個(gè)AbstractFactory基類(lèi)。而這個(gè)Factory類(lèi)就是《設(shè)計(jì)模式》中提到的“子類(lèi)”,它來(lái)決定實(shí)例化那個(gè)類(lèi)。

       下面我們來(lái)具體實(shí)現(xiàn)一下代碼,首先,我們需要兩個(gè)基類(lèi),一個(gè)是Car的,一個(gè)是Factory的。Factory類(lèi)型的作用是構(gòu)建Car的對(duì)象。代碼如下:

public abstract class AbstractCar

    {

        public abstract string Run();

        public abstract string Stop();

        public abstract string Accelerate();

        public abstract string Decelerate();

    }

 

    public abstract class AbstractFactory

    {

        public abstract AbstractCar CreateCar();

    }

    下面,我們來(lái)做一個(gè)BMW的實(shí)現(xiàn)代碼:

public class BMWCar:AbstractCar

    {

        public override string Run()

        {

            return "BMW Run";

        }

 

        public override string Stop()

        {

            return "BMW Stop";

        }

 

        public override string Accelerate()

        {

            return "BMW Accelerate";

        }

 

        public override string Decelerate()

        {

            return "BMW Decalerate";

        }

    }

 

    public class BMWFactory:AbstractFactory

    {

        public override AbstractCar CreateCar()

        {

            return new BMWCar();

        }

    }

這樣我們就可以在客戶端程序得到一個(gè)BMW的實(shí)例,并使用它的方法:

class Class1

    {

        /// <summary>

        /// 應(yīng)用程序的主入口點(diǎn)。

        /// </summary>

        [STAThread]

        static void Main(string[] args)

        {

            AbstractCar car = CreateCarFunc(new BMWFactory ());

            Console.Write(car.Run() + "\n");

            Console.Write(car.Stop() + "\n");

            Console.Write(car.Accelerate() + "\n");

            Console.Write(car.Decelerate() + "\n");

            Console.Read();

        }

 

        public static AbstractCar CreateCarFunc(AbstractFactory factory)

        {

            return factory.CreateCar();

        }

}

在客戶端程序中,我們AbstractFactory的對(duì)象來(lái)得到Car的對(duì)象

結(jié)果如下:

BMW Run

BMW Stop

BMW Accelerate

BMW Decalerate

如果我們需求變了,現(xiàn)在要BORA的對(duì)象,那末,我們首先要對(duì)程序作一下擴(kuò)展,先加入BOARCar類(lèi)和Factory類(lèi),代碼如下:

public class BORACar:AbstractCar

    {

        public override string Run()

        {

            return "BORA Run";

        }

 

        public override string Stop()

        {

            return "BORA Stop";

        }

 

        public override string Accelerate()

        {

            return "BORA Accelerate";

        }

 

        public override string Decelerate()

        {

            return "BORA Decelerate";

        }

    }

 

    public class BORAFactory:AbstractFactory

    {

        public override AbstractCar CreateCar()

        {

            return new BORACar();

        }

}

在客戶端程序中,我們只要稍作修改,將BMWFactory的實(shí)例化變?yōu)?span lang="EN-US">BORAFactory的實(shí)例化就可以,代碼如下:

class Class1

    {

        /// <summary>

        /// 應(yīng)用程序的主入口點(diǎn)。

        /// </summary>

        [STAThread]

        static void Main(string[] args)

        {

            AbstractCar car = CreateCarFunc(new BORAFactory());

            Console.Write(car.Run() + "\n");

            Console.Write(car.Stop() + "\n");

            Console.Write(car.Accelerate() + "\n");

            Console.Write(car.Decelerate() + "\n");

            Console.Read();

        }

 

        public static AbstractCar CreateCarFunc(AbstractFactory factory)

        {

            return factory.CreateCar();

        }

}

得到的結(jié)果是:

BORA Run

BORA Stop

BORA Accelerate

BORA Decelerate

Factory Method的幾個(gè)要點(diǎn):

1、Factory Method模式主要用于隔離類(lèi)對(duì)象的使用者和具體類(lèi)型之間的耦合關(guān)系。面對(duì)一個(gè)經(jīng)常變化的具體類(lèi)型,緊耦合關(guān)系會(huì)導(dǎo)致軟件的脆弱。

2、Factory Method模式通過(guò)面向?qū)ο蟮氖址?,將所要?jiǎng)?chuàng)建的對(duì)象工作延遲到子類(lèi),從而實(shí)現(xiàn)一種擴(kuò)展的策略,較好的解決了這種緊耦合關(guān)系。

posted on 2006-05-15 08:44 KiddLee 閱讀(1496) 評(píng)論(7)  編輯 收藏 引用 網(wǎng)摘 所屬分類(lèi): 設(shè)計(jì)模式

FeedBack:
# 
圖中抽象類(lèi)的名稱(chēng)應(yīng)該用斜體吧?  回復(fù)  更多評(píng)論
  
# re: 設(shè)計(jì)模式學(xué)習(xí)筆記(五)——FactoryMethod工廠模式 2006-05-15 09:38 bisou.cn
一直弄不清楚 抽象工廠 和工廠模式 兩種設(shè)計(jì)模式,學(xué)習(xí)的時(shí)候教程上就說(shuō)兩個(gè)容易混淆,可我還是混淆了。

我感覺(jué) 抽象工廠主要是針對(duì)類(lèi)的
工廠模式 主要針對(duì)類(lèi)里面的方法的!

在實(shí)際應(yīng)用中大多都同時(shí)用到這兩種設(shè)計(jì)模式。比如微軟寫(xiě)的那個(gè)操作數(shù)據(jù)庫(kù)的。

不知道這樣理解對(duì)不?

繼續(xù)關(guān)注中!  回復(fù)  更多評(píng)論
  
# re: 設(shè)計(jì)模式學(xué)習(xí)筆記(五)——FactoryMethod工廠模式 2006-05-15 12:56 Bruce Lee
抽象工廠是解決系列對(duì)象
工廠模式是解決單個(gè)對(duì)象  回復(fù)  更多評(píng)論
  
# re: 設(shè)計(jì)模式學(xué)習(xí)筆記(五)——FactoryMethod工廠模式 2006-05-15 12:56 kid_li
@bisou.cn
謝謝你的提醒,以后畫(huà)圖時(shí),我會(huì)注意的。對(duì)于抽象工廠和工廠方法的區(qū)別我們可以從他們針對(duì)的問(wèn)題來(lái)判斷:
抽象工廠模式:是解決“系列對(duì)象”的需求變化
工廠方法模式:是解決“單個(gè)對(duì)象”的需求變化  回復(fù)  更多評(píng)論
  
# re: 設(shè)計(jì)模式學(xué)習(xí)筆記(五)——FactoryMethod工廠模式 2006-05-16 10:01 bisou.cn
希望樓主繼續(xù)寫(xiě)
更加平易近人的

支持并關(guān)注中  回復(fù)  更多評(píng)論
  
# re: 設(shè)計(jì)模式學(xué)習(xí)筆記(五)——FactoryMethod工廠模式 2006-05-16 10:56 共享精神
看是看懂了,但是不知道具體為SM要這樣寫(xiě),也許以后在項(xiàng)目中應(yīng)用的時(shí)候.
就能明白其中的奧妙!
感謝樓主的好文章!  回復(fù)  更多評(píng)論
  
# re: 設(shè)計(jì)模式學(xué)習(xí)筆記(五)——FactoryMethod工廠模式 2006-09-29 16:10 showmi
好文章,但還沒(méi)看抽象工廠模式  回復(fù)  更多評(píng)論
  

    本站是提供個(gè)人知識(shí)管理的網(wǎng)絡(luò)存儲(chǔ)空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點(diǎn)。請(qǐng)注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購(gòu)買(mǎi)等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)點(diǎn)擊一鍵舉報(bào)。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評(píng)論

    發(fā)表

    請(qǐng)遵守用戶 評(píng)論公約

    類(lèi)似文章 更多