建造者模式(Builder Pattern)
——.NET设计模式系列之四
Terrylee,2005年12月17日
概述
在软件系统中,有时候面临着“一个复杂对象”的创建工作,其通常由各个部分的子对象用一定的算法构成;由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法确相对稳定。如何应对这种变化?如何提供一种“封装机制”来隔离出“复杂对象的各个部分”的变化,从而保持系统中的“稳定构建算法”不随着需求改变而改变?这就是要说的建造者模式。
本文通过现实生活中的买KFC的例子,用图解的方式来诠释建造者模式。
意图
将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示。
模型图
生活中的例子
生成器模式将复杂对象的构建与对象的表现分离开来,这样使得同样的构建过程可以创建出不同的表现。这种模式用于快餐店制作儿童餐。典型的儿童餐包括一个主食,一个辅食,一杯饮料和一个玩具(例如汉堡、炸鸡、可乐和玩具车)。这些在不同的儿童餐中可以是不同的,但是组合成儿童餐的过程是相同的。无论顾客点的是汉堡,三名治还是鸡肉,过程都是一样的。柜台的员工直接把主食,辅食和玩具放在一起。这些是放在一个袋子中的。饮料被倒入杯中,放在袋子外边。这些过程在相互竞争的餐馆中是同样的。
实现过程图解
在这里我们还是以去KFC店买套餐为例子,示意图如下:
客户端:顾客。想去买一套套餐(这里面包括汉堡,可乐,薯条),可以有1号和2号两种套餐供顾客选择。
指导者角色:收银员。知道顾客想要买什么样的套餐,并告诉餐馆员工去准备套餐。
建造者角色:餐馆员工。按照收银员的要求去准备具体的套餐,分别放入汉堡,可乐,薯条等。
产品角色:最后的套餐,所有的东西放在同一个盘子里面。
下面开始我们的买套餐过程。
1.客户创建Derector对象,并用它所想要的Builder对象进行配置。顾客进入KFC店要买套餐,先找到一个收银员,相当于创建了一个指导者对象。这位收银员给出两种套餐供顾客选择:1普通套餐,2黄金套餐。完成的工作如时序图中红色部分所示。
程序实现:
1
usingSystem;
2
usingSystem.Configuration;
3
usingSystem.Reflection;
4
5
namespaceKFC
6
{
7
/**////<summary>
8
///Client类
9
///</summary>
10
publicclassClient
11
{
12
publicstaticvoidMain(string[]args)
13
{
14
FoodManagerfoodmanager=newFoodManager();
15
16
Builderinstance;
17
18
Console.WriteLine("PleaseEnterFoodNo:");
19
20
stringNo=Console.ReadLine();
21
22
stringfoodType=ConfigurationSettings.AppSettings["No"+No];
23
24
instance=(Builder)Assembly.Load("KFC").CreateInstance("KFC."+foodType);
25
26
foodmanager.Construct(instance);
27
}
28
}
29
}
30
产品(套餐)类:
1
usingSystem;
2
usingSystem.Collections;
3
4
namespaceKFC
5
{
6
/**////<summary>
7
///Food类,即产品类
8
///</summary>
9
publicclassFood
10
{
11
Hashtablefood=newHashtable();
12
13
/**////<summary>
14
///添加食品
15
///</summary>
16
///<paramname="strName">食品名称</param>
17
///<paramname="Price">价格</param>
18
publicvoidAdd(stringstrName,stringPrice)
19
{
20
food.Add(strName,Price);
21
}
22
23
/**////<summary>
24
///显示食品清单
25
///</summary>
26
publicvoidShow()
27
{
28
IDictionaryEnumeratormyEnumerator=food.GetEnumerator();
29
Console.WriteLine("FoodList:");
30
Console.WriteLine("------------------------------");
31
stringstrfoodlist="";
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
usingSystem;
2
3
namespaceKFC
4
{
5
/**////<summary>
6
///FoodManager类,即指导者
7
///</summary>
8
publicclassFoodManager
9
{
10
publicvoidConstruct(Builderbuilder)
11
{
12
builder.BuildHamb();
13
14
builder.BuildCoke();
15
16
builder.BuildChip();
17
}
18
}
19
}
20
3.建造者处理指导者的要求,并将部件添加到产品中。餐馆员工(建造者)按照收银员要求的把对应的汉堡,可乐,薯条放入盘子中。这部分是建造者模式里面富于变化的部分,因为顾客选择的套餐不同,套餐的组装过程也不同,这步完成产品对象的创建工作。
程序实现:
1
usingSystem;
2
3
namespaceKFC
4
{
5
/**////<summary>
6
///Builder类,即抽象建造者类,构造套餐
7
///</summary>
8
publicabstractclassBuilder
9
{
10
/**////<summary>
11
///添加汉堡
12
///</summary>
13
publicabstractvoidBuildHamb();
14
15
/**////<summary>
16
///添加可乐
17
///</summary>
18
publicabstractvoidBuildCoke();
19
20
/**////<summary>
21
///添加薯条
22
///</summary>
23
publicabstractvoidBuildChip();
24
25
/**////<summary>
26
///返回结果
27
///</summary>
28
///<returns>食品对象</returns>
29
publicabstractFoodGetFood();
30
}
31
}
32
1
usingSystem;
2
3
namespaceKFC
4
{
5
/**////<summary>
6
///NormalBuilder类,具体构造者,普通套餐
7
///</summary>
8
publicclassNormalBuilder:Builder
9
{
10
privateFoodNormalFood=newFood();
11
12
publicoverridevoidBuildHamb()
13
{
14
NormalFood.Add("NormalHamb","¥10.50");
15
}
16
17
publicoverridevoidBuildCoke()
18
{
19
NormalFood.Add("CokeCole","¥4.50");
20
}
21
22
publicoverridevoidBuildChip()
23
{
24
NormalFood.Add("FireChips","¥2.00");
25
}
26
27
publicoverrideFoodGetFood()
28
{
29
returnNormalFood;
30
}
31
32
}
33
}
34
1
usingSystem;
2
3
namespaceKFC
4
{
5
/**////<summary>
6
///GoldBuilder类,具体构造者,黄金套餐
7
///</summary>
8
publicclassGoldBuilder:Builder
9
{
10
privateFoodGoldFood=newFood();
11
12
publicoverridevoidBuildHamb()
13
{
14
GoldFood.Add("GoldHamb","¥13.50");
15
}
16
17
publicoverridevoidBuildCoke()
18
{
19
GoldFood.Add("CokeCole","¥4.50");
20
}
21
22
publicoverridevoidBuildChip()
23
{
24
GoldFood.Add("FireChips","¥3.50");
25
}
26
27
publicoverrideFoodGetFood()
28
{
29
returnGoldFood;
30
}
31
32
}
33
}
34
4.客户从建造者检索产品。从餐馆员工准备好套餐后,顾客再从餐馆员工那儿拿回套餐。这步客户程序要做的仅仅是取回已经生成的产品对象,如时序图中红色部分所示。
完整的客户程序:
1
usingSystem;
2
usingSystem.Configuration;
3
usingSystem.Reflection;
4
5
namespaceKFC
6
{
7
/**////<summary>
8
///Client类
9
///</summary>
10
publicclassClient
11
{
12
publicstaticvoidMain(string[]args)
13
{
14
FoodManagerfoodmanager=newFoodManager();
15
16
Builderinstance;
17
18
Console.WriteLine("PleaseEnterFoodNo:");
19
20
stringNo=Console.ReadLine();
21
22
stringfoodType=ConfigurationSettings.AppSettings["No"+No];
23
24
instance=(Builder)Assembly.Load("KFC").CreateInstance("KFC."+foodType);
25
26
foodmanager.Construct(instance);
27
28
Foodfood=instance.GetFood();
29
food.Show();
30
31
Console.ReadLine();
32
}
33
}
34
}
35
通过分析不难看出,在这个例子中,在准备套餐的过程是稳定的,即按照一定的步骤去做,而套餐的组成部分则是变化的,有可能是普通套餐或黄金套餐等。这个变化就是建造者模式中的“变化点“,就是我们要封装的部分。
另外一个例子
在这里我们再给出另外一个关于建造房子的例子。客户程序通过调用指导者 (CDirector class)的BuildHouse()方法来创建一个房子。该方法有一个布尔型的参数blnBackyard,当blnBackyard为假时指导者将创建一个Apartment(Concrete Builder),当它为真时将创建一个Single Family Home(Concrete Builder)。这两种房子都实现了接口Ihouse。
程序实现:
1
//关于建造房屋的例子
2
usingSystem;
3
usingSystem.Collections;
4
5
/**////<summary>
6
///抽象建造者
7
///</summary>
8
publicinterfaceIHouse
9
{
10
boolGetBackyard();
11
longNoOfRooms();
12
stringDescription();
13
}
14
15
/**////<summary>
16
///具体建造者
17
///</summary>
18
publicclassCApt:IHouse
19
{
20
privateboolmblnBackyard;
21
privateHashtableRooms;
22
publicCApt()
23
{
24
CRoomroom;
25
Rooms=newHashtable();
26
room=newCRoom();
27
room.RoomName="MasterBedroom";
28
Rooms.Add("room1",room);
29
30
room=newCRoom();
31
room.RoomName="SecondBedroom";
32
Rooms.Add("room2",room);
33
34
room=newCRoom();
35
room.RoomName="LivingRoom";
36
Rooms.Add("room3",room);
37
38
mblnBackyard=false;
39
}
40
41
publicboolGetBackyard()
42
{
43
returnmblnBackyard;
44
}
45
publiclongNoOfRooms()
46
{
47
returnRooms.Count;
48
}
49
publicstringDescription()
50
{
51
IDictionaryEnumeratormyEnumerator=Rooms.GetEnumerator();
52
stringstrDescription;
53
strDescription="ThisisanApartmentwith"+Rooms.Count+"Rooms\n";
54
strDescription=strDescription+"ThisApartmentdoesn'thaveabackyard\n";
55
while(myEnumerator.MoveNext())
56
{
57
strDescription=strDescription+"\n"+myEnumerator.Key+"\t"+((CRoom)myEnumerator.Value).RoomName;
58
}
59
returnstrDescription;
60
}
61
}
62
63
/**////<summary>
64
///具体建造者
65
///</summary>
66
publicclassCSFH:IHouse
67
{
68
privateboolmblnBackyard;
69
privateHashtableRooms;
70
publicCSFH()
71
{
72
CRoomroom;
73
Rooms=newHashtable();
74
75
room=newCRoom();
76
room.RoomName="MasterBedroom";
77
Rooms.Add("room1",room);
78
79
room=newCRoom();
80
room.RoomName="SecondBedroom";
81
Rooms.Add("room2",room);
82
83
room=newCRoom();
84
room.RoomName="ThirdRoom";
85
Rooms.Add("room3",room);
86
87
room=newCRoom();
88
room.RoomName="LivingRoom";
89
Rooms.Add("room4",room);
90
91
room=newCRoom();
92
room.RoomName="GuestRoom";
93
Rooms.Add("room5",room);
94
95
mblnBackyard=true;
96
97
}
98
99
publicboolGetBackyard()
100
{
101
returnmblnBackyard;
102
}
103
publiclongNoOfRooms()
104
{
105
returnRooms.Count;
106
}
107
publicstringDescription()
108
{
109
IDictionaryEnumeratormyEnumerator=Rooms.GetEnumerator();
110
stringstrDescription;
111
strDescription="ThisisanSingleFamilyHomewith"+Rooms.Count+"Rooms\n";
112
strDescription=strDescription+"Thishousehasabackyard\n";
113
while(myEnumerator.MoveNext())
114
{
115
strDescription=strDescription+"\n"+myEnumerator.Key+"\t"+((CRoom)myEnumerator.Value).RoomName;
116
}
117
returnstrDescription;
118
}
119
}
120
121
publicinterfaceIRoom
122
{
123
stringRoomName
{get;set;}
124
}
125
126
publicclassCRoom:IRoom
127
{
128
privatestringmstrRoomName;
129
publicstringRoomName
130
{
131
get
132
{
133
returnmstrRoomName;
134
}
135
set
136
{
137
mstrRoomName=value;
138
}
139
}
140
}
141
142
/**////<summary>
143
///指导者
144
///</summary>
145
publicclassCDirector
146
{
147
publicIHouseBuildHouse(boolblnBackyard)
148
{
149
if(blnBackyard)
150
{
151
returnnewCSFH();
152
}
153
else
154
{
155
returnnewCApt();
156
}
157
}
158
}
159
160
/**////<summary>
161
///客户程序
162
///</summary>
163
publicclassClient
164
{
165
staticvoidMain(string[]args)
166
{
167
CDirectorobjDirector=newCDirector();
168
IHouseobjHouse;
169
170
stringInput=Console.ReadLine();
171
objHouse=objDirector.BuildHouse(bool.Parse(Input));
172
173
Console.WriteLine(objHouse.Description());
174
Console.ReadLine();
175
}
176
}
177
178
建造者模式的几种演化
省略抽象建造者角色
系统中只需要一个具体建造者,省略掉抽象建造者,结构图如下:
指导者代码如下:
1
classDirector
2
{
3
privateConcreteBuilderbuilder;
4
5
publicvoidConstruct()
6
{
7
builder.BuildPartA();
8
builder.BuildPartB();
9
}
10
}
省略指导者角色
抽象建造者角色已经被省略掉,还可以省略掉指导者角色。让Builder角色自己扮演指导者与建造者双重角色。结构图如下:
建造者角色代码如下:
1
publicclassBuilder
2
{
3
privateProductproduct=newProduct();
4
5
publicvoidBuildPartA()
6
{
7
//
8
}
9
10
publicvoidBuildPartB()
11
{
12
//
13
}
14
15
publicProductGetResult()
16
{
17
returnproduct;
18
}
19
20
publicvoidConstruct()
21
{
22
BuildPartA();
23
BuildPartB();
24
}
25
}
客户程序:
1
publicclassClient
2
{
3
privatestaticBuilderbuilder;
4
5
publicstaticvoidMain()
6
{
7
builder=newBuilder();
8
builder.Construct();
9
Productproduct=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》
相关推荐
建造者模式【Builder Pattern】(二)问题改进
建造者模式【Builder Pattern】(一)问题提出
建造者模式【Builder Pattern】(三)问题引申
建造者模式【Builder Pattern】(一)问题改进
3. **提供统一的接口**:客户端(即使用该模式的代码)只需要知道所需的建造者类型,而不需要了解复杂对象的内部组成部分和装配方式。这样,客户端代码就可以专注于高层逻辑,而不是对象的创建细节。 4. **封装构建...
C#设计模式之建造者(Builder)模式示例源代码
设计模式C++学习之建造者模式(Builder)
主要为大家详细介绍了java实现建造者模式Builder Pattern,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
明确建造者(Builder)、具体建造者(Concrete Builder)、指导者(Director)、产品(Product)之间的职责和联系。 ◆建造者(Builder): 给出一个抽象接口,以规范产品对象的各个组成成分的建造。一般而言,...
NULL 博文链接:https://jacky-dai.iteye.com/blog/2295399
NULL 博文链接:https://wy649898543.iteye.com/blog/1431751
设计模式之建造者模式代码示例,
c++实现建造者模式,可直接运行
主要介绍了Java Builder Pattern建造者模式详解及实例的相关资料,需要的朋友可以参考下
10、建造者模式BUILDER PATTERN 11、桥梁模式BRIDGE PATTERN 12、命令模式COMMAND PATTERN 13、装饰模式DECORATOR PATTERN 14、迭代器模式ITERATOR PATTERN 15、组合模式COMPOSITE PATTERN 16、观察者模式...
BuilderPattern.unitypackage是一个建造者模式的例子。
Java 设计模式-建造者模式(Builder)结合Android小知识讲解,简单明了
这个http://blog.csdn.net/dawanganban/article/details/9990405博客中java设计模式的源代码。下载前请先看《设计模式——建造者模式》一文。
建造者模式.ppt 建造者模式可以将部件和其组装过程分开,一步一步创建一个复杂的对象
07 第7章 建造者模式07 第7章 建造者模式07 第7章 建造者模式