设计模式之建造者模式——Builder
一、概述
Builder模式,中文名为建造者模式,又名生成器模式、构建者模式等,是创建型设计模式之一。用于将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。
1.适用性:
- 对象的创建比较复杂、有多种创建形式时
- 创建复杂对象的算法与对象内部组成和装配是相对独立的
2.UML类图
Builder
:定义创建Product
各个部件的抽象接口ConcreteBuilder
:继承自Builder
,实现创建具体类型的Product
所有部件的接口,并提供一个获取最终产品的接口。Director
:借助Builder
,内部封装创建产品的具体流程。
3.UML序列图
- Client创建一个ConcreteBuilder
- Client通过传入ConcreteBuilder创建一个Director
- Client使用Director构造产品
- Client使用ConcreteBuilder组装并获取最终产品
二、实例
想象一下,你作为造物主,你需要创造各种各样的动物,比如狗和猫。查看源码
1.在创造前,要先想好狗和猫大体是什么样子,很显然,它们应该有头、身体和脚。
abstract class Animal
{
public string Head { get; set; }
public string Body { get; set; }
public string Foots { get; set; }
public abstract void Display();
}
class Dog : Animal
{
public override void Display()
{
Console.WriteLine("-------------- 狗的基本信息 -------------------");
Console.WriteLine($"头:{Head}");
Console.WriteLine($"身体:{Body}");
Console.WriteLine($"脚:{Foots}");
}
}
class Cat : Animal
{
public override void Display()
{
Console.WriteLine("-------------- 猫的基本信息 -------------------");
Console.WriteLine($"头:{Head}");
Console.WriteLine($"身体:{Body}");
Console.WriteLine($"脚:{Foots}");
}
}
2.动物的基本特征确定了,也就明确了要干什么活儿了——创建动物的头、身体和脚。
interface IAnimalBuilder
{
void SetHead();
void SetBody();
void SetFoots();
}
3.接下来就是考虑狗和猫各部位的创建细节了。不过,别着急编码,先来考虑一件事情:
DogBuilder
里面除了需要实现IAnimalBuilder
的所有接口外,还需要提供一个GetDog()
的方法,用来把创建好的Dog
给外部;同样的CatBuilder
中也需要提供一个GetCat()
方法。这世间动物有多少种?一千?一万?远远不止!在编码的过程中很容易就忘记提供这个方法了。所以我们在抽象Builder和具体Builder中间插入一个抽象层IAnimalBuilder<TAnimal>
,所有的ConcreteBuilder都继承自这个接口。
interface IAnimalBuilder<TAnimal> : IAnimalBuilder where TAnimal : Animal
{
TAnimal GetAnimal();
}
class DogBuilder : IAnimalBuilder<Dog>
{
private readonly Dog _dog = new Dog();
public void SetHead()
{
_dog.Head = "狗的头";
}
public void SetBody()
{
_dog.Body = "狗的身体";
}
public void SetFoots()
{
_dog.Foots = "狗的脚";
}
public Dog GetAnimal()
{
return _dog;
}
}
class CatBuilder : IAnimalBuilder<Cat>
{
private readonly Cat _cat = new Cat();
public void SetHead()
{
_cat.Head = "猫的头";
}
public void SetBody()
{
_cat.Body = "猫的身体";
}
public void SetFoots()
{
_cat.Foots = "猫的脚";
}
public Cat GetAnimal()
{
return _cat;
}
}
4.最后,只剩Director
了,它来规划Builder要创建哪些东西。
class Director
{
private readonly IAnimalBuilder _builder;
public Director(IAnimalBuilder builder)
{
_builder = builder;
}
public void Construct()
{
_builder.SetHead();
_builder.SetBody();
_builder.SetFoots();
}
}
大功告成!接下来就尝试一下吧
var dogBuilder = new DogBuilder();
var dogDirector = new Director(dogBuilder);
dogDirector.Construct();
dogBuilder.GetAnimal().Display();
var catBuilder = new CatBuilder();
var catDirector = new Director(catBuilder);
catDirector.Construct();
catBuilder.GetAnimal().Display();
三、变种
实际开发过程中,经常会遇到一种需求:创建对象时,对象的一些部分可以自由选择设置或不设置,但是对象一旦创建完成,便不能对其进行任何修改。例如:ASP.NET Core
框架中WebHost
的创建。接下来,我们来模拟一下女娲造人。
1.同样的,我们先设定好人的基本属性
class Person
{
public string Name { get; set; }
public int Gender { get; set; }
public int Age { get; set; }
}
2.接下来,我们来定义IPersonBuilder
接口,除了Build
方法外,我们都返回了IPersonBuilder
,对于习惯于使用Linq
的.Net开发人员来说,再熟悉不过了,它最大的好处就是能够让我们使用Fluent
的方式来编写代码。
public interface IPersonBuilder
{
//注意,这里有问题
Person Build();
IPersonBuilder SetName(string name);
IPersonBuilder SetGender(int gender);
IPersonBuilder SetAge(int age);
}
你发现什么问题了吗?没错,这违背了面向对象五大原则之一——依赖倒置(抽象不应该依赖具体);而且,这也不满足“对象创建后不得进行更改”的需求。所以,我们需要提取出一个抽象层IPersion
。
public interface IPerson
{
string Name { get; }
int Gender { get; }
int Age { get; }
}
class Person : IPerson
{
public string Name { get; set; }
public int Gender { get; set; }
public int Age { get; set; }
}
public interface IPersonBuilder
{
IPerson Build();
IPersonBuilder SetName(string name);
IPersonBuilder SetGender(int gender);
IPersonBuilder SetAge(int age);
}
3.下一步就是构建PersonBuilder
了,用来实现IPersonBuilder
接口的具体逻辑。
class PersonBuilder : IPersonBuilder
{
private readonly Person _person = new Person();
public IPerson Build()
{
return _person;
}
public IPersonBuilder SetName(string name)
{
_person.Name = name;
return this;
}
public IPersonBuilder SetGender(int gender)
{
_person.Gender = gender;
return this;
}
public IPersonBuilder SetAge(int age)
{
_person.Age = age;
return this;
}
}
//提供一个辅助类,用来帮 Client 创建 PersonBuilder
public class PersonBuilderHelper
{
public static IPersonBuilder CreatePersonBuilder()
{
return new PersonBuilder();
}
}
4.Ok,大功告成!来测试一下吧
var person = PersonBuilderHelper.CreatePersonBuilder()
.SetAge(20)
.SetName("jjj")
.Build();
//输出:jjj,0,20
Console.WriteLine($"{person.Name},{person.Gender},{person.Age}");
四、总结
经典版与变种版本的区别:
- 经典版倾向于将构建规划封装起来,Client不关心内部逻辑;而变种版本消除了Director,使得Client拥有更多的主动权,可以自由地进行构建。
- 经典版需要使用Director进行构建,然后使用Builder进行组装返回结果;变种版本则直接使用Builder进行链式构建并组装返回结果,结构更加清晰明了。
- 经典版常用于多种产品的构建是比较固定的情况,Director种类也不宜过多,且必须适应所有产品;而变种版更倾向于某一种产品的构建,构建方式不固定、相当复杂的情况。
查看源码
如果有新发现会进行补充!
设计模式之建造者模式——Builder的更多相关文章
- 乐在其中设计模式(C#) - 建造者模式(Builder Pattern)
原文:乐在其中设计模式(C#) - 建造者模式(Builder Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 建造者模式(Builder Pattern) 作者:webabc ...
- 【设计模式】建造者模式 Builder Pattern
前面学习了简单工厂模式,工厂方法模式以及抽象工厂模式,这些都是创建类的对象所使用的一些常用的方法和套路, 那么如果我们创建一个很复杂的对象可上面的三种方法都不太适合,那么“专业的事交给专业人去做”,2 ...
- 二十四种设计模式:建造者模式(Builder Pattern)
建造者模式(Builder Pattern) 介绍将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. 示例用同样的构建过程创建Sql和Xml的Insert()方法和Get()方 ...
- 设计模式-05建造者模式(Builder Pattern)
1.模式动机 比如我们要组装一台电脑,都知道电脑是由 CPU.主板.内存.硬盘.显卡.机箱.显示器.键盘和鼠标组成,其中非常重要的一点就是这些硬件都是可以灵活选择,但是组装步骤都是大同小异(可以组一个 ...
- 设计模式之建造者模式(Builder)
建造者模式原理:建造模式主要是用于产生对象的各个组成部分,而抽象工厂模式则用于产生一系列对象,建造者模式而且要求这些对象的组成部分有序. 代码如下: #include <iostream> ...
- 设计模式之建造者模式Builder(创建型)
1. 概述 在软件开发的过程中,当遇到一个“复杂的对象”的创建工作,该对象由一定各个部分的子对象用一定的算法构成,由于需求的变化,复杂对象的各个部分经常面临剧烈的变化,但将它们组合在一起的算法相对稳定 ...
- Python 设计模式之建造者模式 Builder Pattern
#引入建造者模式 肯德基的菜单上有 薯条, 鸡腿,鸡翅,鸡米花,可乐,橙汁,火腿汉堡,至尊虾汉堡,牛肉汉堡 , 鸡肉卷等这些单品,也有很多套餐. 比如 套餐1:鸡翅,至尊虾汉堡,可乐,薯条 套餐2:鸡 ...
- 【UE4 设计模式】建造者模式 Builder Pattern
概述 描述 建造者模式,又称生成器模式.是将一个复杂的对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. 建造者模式将客户端与包含多个组成部分的复杂对象的创建过程分离,客户端无需知道复杂 ...
- 创建型设计模式之建造者模式(Builder)
结构 意图 将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示. 适用性 当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式时. 当构造过程必须允许被构造的对象有不 ...
随机推荐
- 微信支付——基于laravel框架的php实现
现在经手的几乎每个项目都支持微信支付,简单记录下接入的大致流程. 1.首先商户等申请各种账号,微信支付商户号,APPID,API密钥,Appsecret 2.app端上传支付需要的各个字段 3.后台收 ...
- 在webconfig放置固定值
通常的,为了布置到服务器后修改的方便通常把一些会改变的值放在webconfig: 首先在web.ocnfig中放入如下值 <appSettings> <add key="A ...
- [POJ1185][NOI2001]炮兵阵地 状压DP
题目链接:http://poj.org/problem?id=1185 很裸的状压,考虑对于一行用二进制储存每一种的状态,但是状态太多了做不了. 观察到有很多状态都是不合法的,于是我们预处理出合法的状 ...
- Android 如何通过Retrofit提交Json格式数据
本文将介绍如何通过retrofit库post一串json格式的数据.首先post的json数据格式如下: { "Id": "string", "Dev ...
- as 开启代码混淆和混淆规则
app的builde.gradle的文件下,buildTypes节点添加release节点,minifyEnabled属性表示是否开启混淆,proguardFiles表示混淆依赖的文件,具体开启方法如 ...
- 线程池 Threadlocal 使用注意
线程池中的线程是重复使用的,即一次使用完后,会被重新放回线程池,可被重新分配使用. 因此,ThreadLocal线程变量,如果保存的信息只是针对一次请求的,放回线程池之前需要清空这些Threadloc ...
- 洛谷 First Step (ファーストステップ) 3月月赛T1
题目背景 知らないことばかりなにもかもが(どうしたらいいの?) 一切的一切 尽是充满了未知数(该如何是好) それでも期待で足が軽いよ(ジャンプだ!) 但我仍因满怀期待而步伐轻盈(起跳吧!) 温度差なん ...
- 洛谷 P2910 [USACO08OPEN]寻宝之路Clear And Present Danger
题目描述 Farmer John is on a boat seeking fabled treasure on one of the N (1 <= N <= 100) islands ...
- 博客高亮代码及使用OpenLiveWriter修改之前博客
简述: 最近查阅前辈资料的时候,看到写的博客很有条理,回头看下自己的乱做麻花,然后来时研究: 他们的代码看起来很漂亮然后我就查资料,在网页版上一直没法出来像他们的格式,后查资料看来的使用客户端工具才 ...
- Java Web应用中获取用户请求相关信息,如:IP地址、操作系统、浏览器等信息
引入jar包 <dependency> <groupId>eu.bitwalker</groupId> <artifactId>UserAgentUti ...