码迷,mamicode.com
首页 > Web开发 > 详细

.NET设计模式(4):建造者模式(Builder Pattern)(转)

时间:2015-03-07 15:35:07      阅读:213      评论:0      收藏:0      [点我收藏+]

标签:

概述

在软件系统中,有时候面临着“一个复杂对象”的创建工作,其通常由各个部分的子对象用一定的算法构成;由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法确相对稳定。如何应对这种变化?如何提供一种“封装机制”来隔离出“复杂对象的各个部分”的变化,从而保持系统中的“稳定构建算法”不随着需求改变而改变?这就是要说的建造者模式。

本文通过现实生活中的买KFC的例子,用图解的方式来诠释建造者模式。

意图

将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示。

模型图

技术分享

生活中的例子

生成器模式将复杂对象的构建与对象的表现分离开来,这样使得同样的构建过程可以创建出不同的表现。这种模式用于快餐店制作儿童餐。典型的儿童餐包括一个主食,一个辅食,一杯饮料和一个玩具(例如汉堡、炸鸡、可乐和玩具车)。这些在不同的儿童餐中可以是不同的,但是组合成儿童餐的过程是相同的。无论顾客点的是汉堡,三名治还是鸡肉,过程都是一样的。柜台的员工直接把主食,辅食和玩具放在一起。这些是放在一个袋子中的。饮料被倒入杯中,放在袋子外边。这些过程在相互竞争的餐馆中是同样的。

技术分享

实现过程图解

在这里我们还是以去KFC店买套餐为例子,示意图如下:

技术分享

客户端:顾客。想去买一套套餐(这里面包括汉堡,可乐,薯条),可以有1号和2号两种套餐供顾客选择。

指导者角色:收银员。知道顾客想要买什么样的套餐,并告诉餐馆员工去准备套餐。

建造者角色:餐馆员工。按照收银员的要求去准备具体的套餐,分别放入汉堡,可乐,薯条等。

产品角色:最后的套餐,所有的东西放在同一个盘子里面。

下面开始我们的买套餐过程。

1.客户创建Derector对象,并用它所想要的Builder对象进行配置。顾客进入KFC店要买套餐,先找到一个收银员,相当于创建了一个指导者对象。这位收银员给出两种套餐供顾客选择:1普通套餐,2黄金套餐。完成的工作如时序图中红色部分所示。

技术分享

程序实现:

 

 1技术分享using System;
 2技术分享using System.Configuration;
 3技术分享using System.Reflection;
 4技术分享
 5技术分享namespace KFC
 6技术分享{
 7技术分享    /// <summary>
 8技术分享    /// Client 类
 9技术分享    /// </summary>

10技术分享    public class Client
11技术分享    {
12技术分享        public static void Main(string[] args)
13技术分享        {
14技术分享            FoodManager foodmanager = new FoodManager();
15技术分享
16技术分享            Builder instance;
17技术分享
18技术分享            Console.WriteLine("Please Enter Food No:");
19技术分享
20技术分享            string No = Console.ReadLine();
21技术分享
22技术分享            string foodType = ConfigurationSettings.AppSettings["No"+No];
23技术分享
24技术分享            instance = (Builder)Assembly.Load("KFC").CreateInstance("KFC." + foodType);
25技术分享
26技术分享            foodmanager.Construct(instance);
27技术分享        }

28技术分享    }

29技术分享}

30技术分享

 

 

产品(套餐)类:

 1技术分享using System;
 2技术分享using System.Collections;
 3技术分享
 4技术分享namespace KFC
 5技术分享{
 6技术分享    /// <summary>
 7技术分享    /// Food类,即产品类
 8技术分享    /// </summary>

 9技术分享    public class Food
10技术分享    {
11技术分享        Hashtable food = new Hashtable();
12技术分享        
13技术分享        /// <summary>
14技术分享        /// 添加食品
15技术分享        /// </summary>
16技术分享        /// <param name="strName">食品名称</param>
17技术分享        /// <param name="Price">价格</param>

18技术分享        public void Add(string strName,string Price)
19技术分享        {
20技术分享            food.Add(strName,Price);
21技术分享        }

22技术分享        
23技术分享        /// <summary>
24技术分享        /// 显示食品清单
25技术分享        /// </summary>

26技术分享        public void Show()
27技术分享        {
28技术分享            IDictionaryEnumerator myEnumerator  = food.GetEnumerator();
29技术分享            Console.WriteLine("Food List:");
30技术分享            Console.WriteLine("------------------------------");
31技术分享            string strfoodlist = "";
32技术分享            while(myEnumerator.MoveNext())
33技术分享            {
34技术分享                strfoodlist = strfoodlist + "\n\n" + myEnumerator.Key.ToString();
35技术分享                strfoodlist = strfoodlist + ":\t" +myEnumerator.Value.ToString();
36技术分享            }

37技术分享            Console.WriteLine(strfoodlist);
38技术分享            Console.WriteLine("\n------------------------------");
39技术分享        }

40技术分享    }

41技术分享}

42技术分享

2.指导者通知建造器。收银员(指导者)告知餐馆员工准备套餐。这里我们准备套餐的顺序是:放入汉堡,可乐倒入杯中,薯条放入盒中,并把这些东西都放在盘子上。这个过程对于普通套餐和黄金套餐来说都是一样的,不同的是它们的汉堡,可乐,薯条价格不同而已。如时序图红色部分所示:

技术分享

程序实现:

 

 1技术分享using System;
 2技术分享
 3技术分享namespace KFC
 4技术分享{
 5技术分享    /// <summary>
 6技术分享    /// FoodManager类,即指导者
 7技术分享    /// </summary>

 8技术分享    public class FoodManager
 9技术分享    {
10技术分享        public void Construct(Builder builder)
11技术分享        {
12技术分享            builder.BuildHamb();
13技术分享
14技术分享            builder.BuildCoke();
15技术分享
16技术分享            builder.BuildChip();
17技术分享        }
    
18技术分享    }

19技术分享}

20技术分享

 

3.建造者处理指导者的要求,并将部件添加到产品中。餐馆员工(建造者)按照收银员要求的把对应的汉堡,可乐,薯条放入盘子中。这部分是建造者模式里面富于变化的部分,因为顾客选择的套餐不同,套餐的组装过程也不同,这步完成产品对象的创建工作。

程序实现:

 

 1技术分享using System;
 2技术分享
 3技术分享namespace KFC
 4技术分享{
 5技术分享    /// <summary>
 6技术分享    /// Builder类,即抽象建造者类,构造套餐
 7技术分享    /// </summary>

 8技术分享    public abstract class Builder
 9技术分享    {    
10技术分享        /// <summary>
11技术分享        /// 添加汉堡
12技术分享        /// </summary>

13技术分享        public abstract void BuildHamb();
14技术分享        
15技术分享        /// <summary>
16技术分享        /// 添加可乐
17技术分享        /// </summary>

18技术分享        public abstract void BuildCoke();
19技术分享        
20技术分享        /// <summary>
21技术分享        /// 添加薯条
22技术分享        /// </summary>

23技术分享        public abstract void BuildChip();
24技术分享        
25技术分享        /// <summary>
26技术分享        /// 返回结果
27技术分享        /// </summary>
28技术分享        /// <returns>食品对象</returns>

29技术分享        public abstract Food GetFood();
30技术分享    }

31技术分享}

32技术分享

 

 1技术分享using System;
 2技术分享
 3技术分享namespace KFC
 4技术分享{
 5技术分享    /// <summary>
 6技术分享    /// NormalBuilder类,具体构造者,普通套餐
 7技术分享    /// </summary>

 8技术分享    public class NormalBuilder:Builder
 9技术分享    {
10技术分享        private Food NormalFood = new Food();
11技术分享
12技术分享        public override void BuildHamb()
13技术分享        {
14技术分享            NormalFood.Add("NormalHamb","¥10.50");
15技术分享        }

16技术分享        
17技术分享        public override void BuildCoke()
18技术分享        {
19技术分享            NormalFood.Add("CokeCole","¥4.50");
20技术分享        }

21技术分享
22技术分享        public override void BuildChip()
23技术分享        {
24技术分享            NormalFood.Add("FireChips","¥2.00");
25技术分享        }

26技术分享
27技术分享        public override Food GetFood()
28技术分享        {
29技术分享            return NormalFood;
30技术分享        }

31技术分享
32技术分享    }

33技术分享}

34技术分享

 

 1技术分享using System;
 2技术分享
 3技术分享namespace KFC
 4技术分享{
 5技术分享    /// <summary>
 6技术分享    /// GoldBuilder类,具体构造者,黄金套餐
 7技术分享    /// </summary>

 8技术分享    public class GoldBuilder:Builder
 9技术分享    {
10技术分享        private Food GoldFood = new Food();
11技术分享
12技术分享        public override void BuildHamb()
13技术分享        {
14技术分享            GoldFood.Add("GoldHamb","¥13.50");
15技术分享        }

16技术分享        
17技术分享        public override void BuildCoke()
18技术分享        {
19技术分享            GoldFood.Add("CokeCole","¥4.50");
20技术分享        }

21技术分享
22技术分享        public override void BuildChip()
23技术分享        {
24技术分享            GoldFood.Add("FireChips","¥3.50");
25技术分享        }

26技术分享
27技术分享        public override Food GetFood()
28技术分享        {
29技术分享            return GoldFood;
30技术分享        }

31技术分享
32技术分享    }

33技术分享}

34技术分享

 

 

4.客户从建造者检索产品。从餐馆员工准备好套餐后,顾客再从餐馆员工那儿拿回套餐。这步客户程序要做的仅仅是取回已经生成的产品对象,如时序图中红色部分所示。

技术分享

完整的客户程序:

 

 1技术分享using System;
 2技术分享using System.Configuration;
 3技术分享using System.Reflection;
 4技术分享
 5技术分享namespace KFC
 6技术分享{
 7技术分享    /// <summary>
 8技术分享    /// Client 类
 9技术分享    /// </summary>

10技术分享    public class Client
11技术分享    {
12技术分享        public static void Main(string[] args)
13技术分享        {
14技术分享            FoodManager foodmanager = new FoodManager();
15技术分享
16技术分享            Builder instance;
17技术分享
18技术分享            Console.WriteLine("Please Enter Food No:");
19技术分享
20技术分享            string No = Console.ReadLine();
21技术分享
22技术分享            string foodType = ConfigurationSettings.AppSettings["No"+No];
23技术分享
24技术分享            instance = (Builder)Assembly.Load("KFC").CreateInstance("KFC." + foodType);
25技术分享
26技术分享            foodmanager.Construct(instance);
27技术分享
28技术分享            Food food = instance.GetFood();
29技术分享            food.Show();
30技术分享
31技术分享            Console.ReadLine();
32技术分享        }

33技术分享    }

34技术分享}

35技术分享

 

通过分析不难看出,在这个例子中,在准备套餐的过程是稳定的,即按照一定的步骤去做,而套餐的组成部分则是变化的,有可能是普通套餐或黄金套餐等。这个变化就是建造者模式中的“变化点“,就是我们要封装的部分。

另外一个例子

在这里我们再给出另外一个关于建造房子的例子。客户程序通过调用指导者 (CDirector class)BuildHouse()方法来创建一个房子。该方法有一个布尔型的参数blnBackyard,当blnBackyard为假时指导者将创建一个ApartmentConcrete Builder),当它为真时将创建一个Single Family HomeConcrete Builder)。这两种房子都实现了接口Ihouse

程序实现:

 

  1技术分享//关于建造房屋的例子
  2技术分享using System;
  3技术分享using System.Collections;
  4技术分享
  5技术分享/// <summary>
  6技术分享/// 抽象建造者
  7技术分享/// </summary>

  8技术分享public interface IHouse
  9技术分享{
 10技术分享    bool GetBackyard();
 11技术分享    long NoOfRooms();
 12技术分享    string  Description();
 13技术分享}

 14技术分享
 15技术分享/// <summary>
 16技术分享/// 具体建造者
 17技术分享/// </summary>

 18技术分享public class CApt:IHouse
 19技术分享{
 20技术分享    private bool mblnBackyard;
 21技术分享    private Hashtable Rooms;
 22技术分享    public CApt()
 23技术分享    {
 24技术分享        CRoom room;    
 25技术分享        Rooms = new Hashtable();
 26技术分享        room = new CRoom();
 27技术分享        room.RoomName = "Master Bedroom";
 28技术分享        Rooms.Add ("room1",room);
 29技术分享
 30技术分享        room = new CRoom();
 31技术分享        room.RoomName = "Second Bedroom";
 32技术分享        Rooms.Add ("room2",room);
 33技术分享
 34技术分享        room = new CRoom();
 35技术分享        room.RoomName = "Living Room";
 36技术分享        Rooms.Add ("room3",room);
 37技术分享        
 38技术分享        mblnBackyard = false;
 39技术分享    }

 40技术分享
 41技术分享    public bool GetBackyard()
 42技术分享    {
 43技术分享        return mblnBackyard;
 44技术分享    }

 45技术分享    public long NoOfRooms()
 46技术分享    {
 47技术分享        return Rooms.Count; 
 48技术分享    }

 49技术分享    public string  Description()
 50技术分享    {
 51技术分享        IDictionaryEnumerator myEnumerator  = Rooms.GetEnumerator();
 52技术分享        string strDescription;
 53技术分享        strDescription = "This is an Apartment with " + Rooms.Count + " Rooms \n";
 54技术分享        strDescription = strDescription + "This Apartment doesn‘t have a backyard \n";                        
 55技术分享        while (myEnumerator.MoveNext())
 56技术分享        {
 57技术分享            strDescription = strDescription + "\n" + myEnumerator.Key + "\t" + ((CRoom)myEnumerator.Value).RoomName;
 58技术分享        }

 59技术分享        return strDescription;
 60技术分享    }

 61技术分享}

 62技术分享
 63技术分享/// <summary>
 64技术分享/// 具体建造者
 65技术分享/// </summary>

 66技术分享public class CSFH:IHouse
 67技术分享{
 68技术分享    private bool mblnBackyard;
 69技术分享    private Hashtable Rooms;
 70技术分享    public CSFH()
 71技术分享    {
 72技术分享        CRoom room;
 73技术分享        Rooms = new Hashtable();
 74技术分享
 75技术分享        room = new CRoom();
 76技术分享        room.RoomName = "Master Bedroom";
 77技术分享        Rooms.Add ("room1",room);
 78技术分享
 79技术分享        room = new CRoom();
 80技术分享        room.RoomName = "Second Bedroom";
 81技术分享        Rooms.Add ("room2",room);
 82技术分享
 83技术分享        room = new CRoom();
 84技术分享        room.RoomName = "Third Room";
 85技术分享        Rooms.Add ("room3",room);
 86技术分享        
 87技术分享        room = new CRoom();
 88技术分享        room.RoomName = "Living Room";
 89技术分享        Rooms.Add ("room4",room);
 90技术分享
 91技术分享        room = new CRoom();
 92技术分享        room.RoomName = "Guest Room";
 93技术分享        Rooms.Add ("room5",room);
 94技术分享
 95技术分享        mblnBackyard = true;
 96技术分享 
 97技术分享    }

 98技术分享
 99技术分享    public bool GetBackyard()
100技术分享    {
101技术分享        return mblnBackyard;
102技术分享    }

103技术分享    public long NoOfRooms()
104技术分享    {
105技术分享        return Rooms.Count;
106技术分享    }

107技术分享    public string  Description()
108技术分享    {
109技术分享        IDictionaryEnumerator myEnumerator  = Rooms.GetEnumerator();
110技术分享        string strDescription;
111技术分享        strDescription = "This is an Single Family Home with " + Rooms.Count + " Rooms \n";
112技术分享        strDescription = strDescription + "This house has a backyard \n"
113技术分享        while (myEnumerator.MoveNext())
114技术分享        {
115技术分享            strDescription = strDescription + "\n" + myEnumerator.Key + "\t" + ((CRoom)myEnumerator.Value).RoomName; 
116技术分享        }
      
117技术分享        return strDescription;
118技术分享    }

119技术分享}

120技术分享
121技术分享public interface IRoom
122技术分享{
123技术分享    string RoomName{get;set;}
124技术分享}

125技术分享
126技术分享public class CRoom:IRoom
127技术分享{
128技术分享    private string mstrRoomName;
129技术分享    public string RoomName
130技术分享    {
131技术分享        get
132技术分享        {
133技术分享            return mstrRoomName;
134技术分享        }

135技术分享        set 
136技术分享        {
137技术分享            mstrRoomName = value;
138技术分享        }

139技术分享    }

140技术分享}

141技术分享
142技术分享/// <summary>
143技术分享/// 指导者
144技术分享/// </summary>

145技术分享public class CDirector
146技术分享{
147技术分享    public IHouse BuildHouse(bool blnBackyard)
148技术分享    {
149技术分享        if (blnBackyard)
150技术分享        {
151技术分享            return new CSFH();
152技术分享        }

153技术分享        else
154技术分享        {
155技术分享            return new CApt(); 
156技术分享        }

157技术分享    }

158技术分享}

159技术分享
160技术分享/// <summary>
161技术分享/// 客户程序
162技术分享/// </summary>

163技术分享public class Client
164技术分享{
165技术分享    static void Main(string[] args) 
166技术分享    {
167技术分享        CDirector objDirector = new CDirector();
168技术分享        IHouse objHouse;
169技术分享
170技术分享        string Input = Console.ReadLine();
171技术分享        objHouse = objDirector.BuildHouse(bool.Parse(Input));
172技术分享    
173技术分享        Console.WriteLine(objHouse.Description());
174技术分享        Console.ReadLine();
175技术分享    }

176技术分享}

177技术分享
178技术分享

 

建造者模式的几种演化

省略抽象建造者角色

系统中只需要一个具体建造者,省略掉抽象建造者,结构图如下:

技术分享

指导者代码如下:

 

 1技术分享 class Director
 2技术分享  {
 3技术分享   private ConcreteBuilder builder;
 4技术分享 
 5技术分享   public void Construct()
 6技术分享    {
 7技术分享     builder.BuildPartA();
 8技术分享     builder.BuildPartB();
 9技术分享   }

10技术分享 }

 

省略指导者角色

抽象建造者角色已经被省略掉,还可以省略掉指导者角色。让Builder角色自己扮演指导者与建造者双重角色。结构图如下:

技术分享

建造者角色代码如下:

 

 1技术分享 public class Builder
 2技术分享  {
 3技术分享   private Product product = new Product();
 4技术分享 
 5技术分享   public void BuildPartA()
 6技术分享    
 7技术分享     //技术分享技术分享
 8技术分享   }

 9技术分享 
10技术分享   public void BuildPartB()
11技术分享    {
12技术分享     //技术分享技术分享
13技术分享   }

14技术分享 
15技术分享   public Product GetResult()
16技术分享    {
17技术分享     return product;
18技术分享   }

19技术分享 
20技术分享   public void Construct()
21技术分享    {
22技术分享     BuildPartA();
23技术分享     BuildPartB();
24技术分享   }

25技术分享 }


客户程序:

 1技术分享 public class Client
 2技术分享  {
 3技术分享   private static Builder builder;
 4技术分享 
 5技术分享   public static void Main()
 6技术分享    {
 7技术分享     builder = new Builder();
 8技术分享     builder.Construct();
 9技术分享     Product product = builder.GetResult();
10技术分享   }

11技术分享 }

 

 

合并建造者角色和产品角色

建造模式失去抽象建造者角色和指导者角色后,可以进一步退化,从而失去具体建造者角色,此时具体建造者角色和产品角色合并,从而使得产品自己就是自己的建造者。这样做混淆了对象的建造者和对象本身,但是有时候一个产品对象有着固定的几个零件,而且永远只有这几个零件,此时将产品类和建造类合并,可以使系统简单易读。结构图如下:

技术分享

实现要点

1、建造者模式主要用于“分步骤构建一个复杂的对象”,在这其中“分步骤”是一个稳定的算法,而复杂对象的各个部分则经常变化。

2产品不需要抽象类,特别是由于创建对象的算法复杂而导致使用此模式的情况下或者此模式应用于产品的生成过程,其最终结果可能差异很大,不大可能提炼出一个抽象产品类。
3
、创建者中的创建子部件的接口方法不是抽象方法而是空方法,不进行任何操作,具体的创建者只需要覆盖需要的方法就可以,但是这也不是绝对的,特别是类似文本转换这种情况下,缺省的方法将输入原封不动的输出是合理的缺省操作。

4、前面我们说过的抽象工厂模式(Abtract Factory)解决“系列对象”的需求变化,Builder模式解决“对象部分”的需求变化,建造者模式常和组合模式(Composite Pattern)结合使用。

效果

1、建造者模式的使用使得产品的内部表象可以独立的变化。使用建造者模式可以使客户端不必知道产品内部组成的细节。
2
、每一个Builder都相对独立,而与其它的Builder无关。
3
、可使对构造过程更加精细控制。

4、将构建代码和表示代码分开。

5、建造者模式的缺点在于难于应付“分步骤构建算法”的需求变动。

适用性

以下情况应当使用建造者模式:

1、需要生成的产品对象有复杂的内部结构。
2
、需要生成的产品对象的属性相互依赖,建造者模式可以强迫生成顺序。
3
 在对象创建过程中会使用到系统中的一些其它对象,这些对象在产品对象的创建过程中不易得到。

应用场景

1、   RTF文档交换格式阅读器。

2、   .NET环境下的字符串处理StringBuilder,这是一种简化了的建造者模式。

3、   ……

总结

建造者模式的实质是解耦组装过程和创建具体部件,使得我们不用去关心每个部件是如何组装的。

______________________________________________________________________________________

源程序下载:/Files/Terrylee/BuilderPattern.rar

参考资料:

Java与设计模式》阎宏

《设计模式(中文版)》

DesignPatternsExplained

.NET设计模式(4):建造者模式(Builder Pattern)(转)

标签:

原文地址:http://www.cnblogs.com/sandea/p/4320354.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!