.Net創(chuàng)建型設(shè)計(jì)模式之建造者、生成器模式(Builder)
一、動(dòng)機(jī)(Motivation)
在軟件系統(tǒng)中,有時(shí)候面臨著“一個(gè)復(fù)雜對(duì)象”的創(chuàng)建工作,其通常由各個(gè)部分的子對(duì)象用一定的算法構(gòu)成;由于需求的變化,這個(gè)復(fù)雜對(duì)象的各個(gè)部分經(jīng)常面臨著劇烈的變化,但是將它們組合在一起的算法卻相對(duì)穩(wěn)定。
如何應(yīng)對(duì)這種變化?如何提供一種“封裝機(jī)制”來(lái)隔離出“復(fù)雜對(duì)象的各個(gè)部分”的變化,從而保持系統(tǒng)中的“穩(wěn)定構(gòu)建算法”不隨著需求改變而改變?
二、意圖(Intent)
將一個(gè)復(fù)雜對(duì)象的構(gòu)建與其表示相分離,使得同樣的構(gòu)建過(guò)程可以創(chuàng)建不同的表示。——《設(shè)計(jì)模式》GoF
三、結(jié)構(gòu)(Structure)
協(xié)作(Collaborations)
四、模式的組成
(1)、抽象建造者角色(Builder):為創(chuàng)建一個(gè)Product對(duì)象的各個(gè)部件指定抽象接口,以規(guī)范產(chǎn)品對(duì)象的各個(gè)組成成分的建造。一般而言,此角色規(guī)定要實(shí)現(xiàn)復(fù)雜對(duì)象的哪些部分的創(chuàng)建,并不涉及具體的對(duì)象部件的創(chuàng)建。
(2)、具體建造者(ConcreteBuilder)
1)實(shí)現(xiàn)Builder的接口以構(gòu)造和裝配該產(chǎn)品的各個(gè)部件。即實(shí)現(xiàn)抽象建造者角色Builder的方法。
2)定義并明確它所創(chuàng)建的表示,即針對(duì)不同的商業(yè)邏輯,具體化復(fù)雜對(duì)象的各部分的創(chuàng)建
3) 提供一個(gè)檢索產(chǎn)品的接口
4) 構(gòu)造一個(gè)使用Builder接口的對(duì)象即在指導(dǎo)者的調(diào)用下創(chuàng)建產(chǎn)品實(shí)例
(3)、指導(dǎo)者(Director):調(diào)用具體建造者角色以創(chuàng)建產(chǎn)品對(duì)象的各個(gè)部分。指導(dǎo)者并沒(méi)有涉及具體產(chǎn)品類的信息,真正擁有具體產(chǎn)品的信息是具體建造者對(duì)象。它只負(fù)責(zé)保證對(duì)象各部分完整創(chuàng)建或按某種順序創(chuàng)建。
(4)、產(chǎn)品角色(Product):建造中的復(fù)雜對(duì)象。它要包含那些定義組件的類,包括將這些組件裝配成產(chǎn)品的接口。
五、建筑者模式的具體實(shí)現(xiàn)
現(xiàn)在人們的生活水平都提高了,有錢了,我今天就以汽車組裝為例子。每臺(tái)汽車的組裝過(guò)程都是一致的,所以我們使用同樣的構(gòu)建過(guò)程可以創(chuàng)建不同的表示(即可以組裝成不同型號(hào)的汽車,不能像例子這樣,一會(huì)別克,一會(huì)奧迪的)
組裝汽車、電腦、手機(jī)、電視等等負(fù)責(zé)對(duì)象的這些場(chǎng)景都可以應(yīng)用建造者模式來(lái)設(shè)計(jì)。
static void Main(string[] args) { Director director = new Director(); Builder buickCarBuilder = new BuickBuilder(); Builder aoDiCarBuilder = new AoDiBuilder(); director.Construct(buickCarBuilder); Car buickCar = buickCarBuilder.GetCar(); //組裝完成,我來(lái)駕駛別克了 buickCar.Show(); director.Construct(aoDiCarBuilder); // 我老婆就要奧迪了,她比較喜歡大品牌 Car aoDiCar = aoDiCarBuilder.GetCar(); aoDiCar.Show(); } /// <summary> /// 這個(gè)類型才是組裝的,Construct方法里面的實(shí)現(xiàn)就是創(chuàng)建復(fù)雜對(duì)象固定算法的實(shí)現(xiàn),該算法是固定的,或者說(shuō)是相對(duì)穩(wěn)定的 /// 這個(gè)人當(dāng)然就是老板了,也就是建造者模式中的指揮者 /// </summary> public class Director { // 組裝汽車 public void Construct(Builder builder) { builder.BuildCarDoor(); builder.BuildCarWheel(); builder.BuildCarEngine(); } } /// <summary> /// 汽車類 /// </summary> public sealed class Car { // 汽車部件集合 private IList<string> parts = new List<string>(); // 把單個(gè)部件添加到汽車部件集合中 public void Add(string part) { parts.Add(part); } public void Show() { Console.WriteLine("汽車開(kāi)始在組裝......."); foreach (string part in parts) { Console.WriteLine("組件" + part + "已裝好"); } Console.WriteLine("汽車組裝好了"); } } /// <summary> /// 抽象建造者,它定義了要?jiǎng)?chuàng)建什么部件和最后創(chuàng)建的結(jié)果,但是不是組裝的的類型,切記 /// </summary> public abstract class Builder { // 創(chuàng)建車門 public abstract void BuildCarDoor(); // 創(chuàng)建車輪 public abstract void BuildCarWheel(); //創(chuàng)建車引擎 public abstract void BuildCarEngine(); // 獲得組裝好的汽車 public abstract Car GetCar(); } /// <summary> /// 具體創(chuàng)建者,具體的車型的創(chuàng)建者,例如:別克 /// </summary> public sealed class BuickBuilder : Builder { Car buickCar = new Car(); public override void BuildCarDoor() { buickCar.Add("Buick's Door"); } public override void BuildCarWheel() { buickCar.Add("Buick's Wheel"); } public override void BuildCarEngine() { buickCar.Add("Buick's Engine"); } public override Car GetCar() { return buickCar; } } /// <summary> /// 具體創(chuàng)建者,具體的車型的創(chuàng)建者,例如:奧迪 /// </summary> public sealed class AoDiBuilder : Builder { Car aoDiCar = new Car(); public override void BuildCarDoor() { aoDiCar.Add("Aodi's Door"); } public override void BuildCarWheel() { aoDiCar.Add("Aodi's Wheel"); } public override void BuildCarEngine() { aoDiCar.Add("Aodi's Engine"); } public override Car GetCar() { return aoDiCar; } }
六、建造者模式的實(shí)現(xiàn)要點(diǎn)
在建造者模式中,指揮者是直接與客戶端打交道的,指揮者將客戶端創(chuàng)建產(chǎn)品的請(qǐng)求劃分為對(duì)各個(gè)部件的建造請(qǐng)求,再將這些請(qǐng)求委派到具體建造者角色,具體建造者角色是完成具體產(chǎn)品的構(gòu)建工作的,卻不為客戶所知道。建造者模式主要用于“分步驟來(lái)構(gòu)建一個(gè)復(fù)雜的對(duì)象”,其中“分步驟”是一個(gè)固定的組合過(guò)程,而復(fù)雜對(duì)象的各個(gè)部分是經(jīng)常變化的。 產(chǎn)品不需要抽象類,由于建造模式的創(chuàng)建出來(lái)的最終產(chǎn)品可能差異很大,所以不大可能提煉出一個(gè)抽象產(chǎn)品類。
在前面文章中介紹的抽象工廠模式解決了“系列產(chǎn)品”的需求變化,而建造者模式解決的是 “產(chǎn)品部分” 的需要變化。
由于建造者隱藏了具體產(chǎn)品的組裝過(guò)程,所以要改變一個(gè)產(chǎn)品的內(nèi)部表示,只需要再實(shí)現(xiàn)一個(gè)具體的建造者就可以了,從而能很好地應(yīng)對(duì)產(chǎn)品組成組件的需求變化。
1、建造者模式的優(yōu)點(diǎn):
(1)、使用建造者模式可以使客戶端不必知道產(chǎn)品內(nèi)部組成的細(xì)節(jié)。
(2)、具體的建造者類之間是相互獨(dú)立的,容易擴(kuò)展。
(3)、由于具體的建造者是獨(dú)立的,因此可以對(duì)建造過(guò)程逐步細(xì)化,而不對(duì)其他的模塊產(chǎn)生任何影響。
2、建造者模式的缺點(diǎn):
產(chǎn)生多余的Build對(duì)象以及Dirextor類。
3、創(chuàng)建者模式的使用場(chǎng)景:
(1)、當(dāng)創(chuàng)建復(fù)雜對(duì)象的算法應(yīng)該獨(dú)立于該對(duì)象的組成部分以及它們的裝配方式時(shí)。
(2)、相同的方法,不同的執(zhí)行順序,產(chǎn)生不同的事件結(jié)果時(shí)。
(3)、多個(gè)部件或零件,都可以裝配到一個(gè)對(duì)象中,但是產(chǎn)生的運(yùn)行結(jié)果又不相同時(shí)。
(4)、產(chǎn)品類非常復(fù)雜,或者產(chǎn)品類中的調(diào)用順序不同產(chǎn)生了不同的效能。
(5)、創(chuàng)建一些復(fù)雜的對(duì)象時(shí),這些對(duì)象的內(nèi)部組成構(gòu)件間的建造順序是穩(wěn)定的,但是對(duì)象的內(nèi)部組成構(gòu)件面臨著復(fù)雜的變化。
七、.NET框架中的Builder應(yīng)用
在ASP.Net中,我們?cè)趯懸粋€(gè)Page類時(shí),這個(gè)類繼承自System.Web.UI.Page。Page其實(shí)就是一個(gè)Builder,它是一個(gè)容器。它有很多方法,就是所謂的BuilderPart()方法,例如:OnInit()、OnLoad()、OnPreRender()、Render()等,它們都是虛方法,我們都可以去重寫,提供我們自己的實(shí)現(xiàn)。當(dāng)我們編譯到bin文件夾下的dll時(shí),我們就生成了一個(gè)我們自己的ConcreteBuilder實(shí)例。實(shí)際上系統(tǒng)使用的是Page基類。
到此這篇關(guān)于.Net設(shè)計(jì)模式之建造者、生成器模式(Builder)的文章就介紹到這了。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
ASP.NET實(shí)現(xiàn)word文檔在線預(yù)覽功能代碼
文檔管理系統(tǒng)需要實(shí)現(xiàn)WORD能在線預(yù)覽功能,以前覺(jué)得挺難的,經(jīng)過(guò)一番研究發(fā)現(xiàn),WORD自帶的另存為可以保存為HTMl文件。2010-07-07Effective C# 使用成員初始化器而不是賦值語(yǔ)句
一般情況下,一個(gè)類都會(huì)有多個(gè)構(gòu)造函數(shù)。隨著時(shí)間的推移,成員變量、構(gòu)造函數(shù)不斷增加2012-11-11c#中實(shí)現(xiàn)文件拖放打開(kāi)的方法
向ListBox拖入一個(gè)文件,ListBox顯示該文件的路徑,然后單擊該路徑,點(diǎn)擊Open按鈕打開(kāi)該文件2006-10-10部署ASP.NET?Core程序到Windows系統(tǒng)
這篇文章介紹了部署ASP.NET?Core程序到Windows系統(tǒng)的方法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-03-03.Net結(jié)構(gòu)型設(shè)計(jì)模式之橋接模式(Bridge)
這篇文章介紹了.Net結(jié)構(gòu)型設(shè)計(jì)模式之橋接模式(Bridge),文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05