1.建造者模式介绍

  在软件开发中,有时我们要创建一个复杂的对象,这个对象由几个子部件按一定的步骤组合而成,这时候我们就可以使用建造者模式了。说到建造者我们首先想到的是盖房子,盖房子简单的说有三个步骤:打地基,砌砖,粉刷。我们就以盖房子为例解释建造者模式的用法。

  建造者模式有三个角色:建造者,具体的建造者,监工。理清这三个角色的作用我们就可以愉快的使用建造者模式了。

  建造者:一般为抽象类或接口,定义了建造者的功能。如盖房子例子的建造者有打地基,砌砖和粉刷的功能。

  具体的建造者:实现了建造者的抽象方法(或接口)。不同的具体建造者生产的组件不同,如一个技术好的建造者打地基深,砌砖整齐,粉刷光滑,而技术差的建造者打地基浅,砌砖错乱,粉刷粗糙。

  监工:制定建造的算法。建造者可以打地基,砌砖,粉刷,但是不知道先粉刷还是先打地基,监工就是给建造者制定盖房子步骤的。

代码实现如下,

建造者和具体建造者:

   //建造者抽象类,定义了建造者的能力
public abstract class Builder
{
public abstract void Dadiji();//打地基
public abstract void QiZhuan();//砌砖
public abstract void FenShua();//粉刷
} /// <summary>
/// 技术好的建造者
/// </summary>
public class GoodBuilder : Builder
{
private StringBuilder house = new StringBuilder();
public override void Dadiji()
{
house.Append("深地基-->");
//这里一般是new一个部件,添加到实例中,如 house.Diji=new Diji("深地基")
//为了演示方便 用sringBuilder表示一个复杂的房子,string表示房子的部件
} public override void FenShua()
{
house.Append("粉刷光滑-->");
} public override void QiZhuan()
{
house.Append("砌砖整齐-->");
}
public string GetHouse()
{
return house.Append("好质量房子建成了!").ToString();
}
} /// <summary>
/// 技术差的建造者
/// </summary>
public class BadBuilder:Builder
{
private StringBuilder house = new StringBuilder();
public override void Dadiji()
{
house.Append("挖浅地基-->");
} public override void FenShua()
{
house.Append("粉刷粗糙-->");
} public override void QiZhuan()
{
house.Append("砌砖错乱-->");
}
public string GetHouse()
{
return house.Append("坏质量房子建成了!").ToString();
}
}

监工:

    //监工类,制定盖房子的步骤
public class Director
{
private Builder builder;
public Director(Builder builder)
{
this.builder = builder;
} //制定盖房子的流程,
public void Construct()
{
builder.Dadiji();//先打地基
builder.QiZhuan();//再砌砖
builder.FenShua();//最后粉刷
}
}

客户端调用:

    class Program
{
static void Main(string[] args)
{
//监工1派遣技术好的建造者盖房子
GoodBuilder goodBuilder = new GoodBuilder();
Director director1 = new Director(goodBuilder);
director1.Construct();
string house1 = goodBuilder.GetHouse();
Console.WriteLine(house1); //监工2派遣技术差的建造者盖房子
GoodBuilder badBuilder = new GoodBuilder();
Director director2 = new Director(goodBuilder);
director2.Construct();
string house2 = goodBuilder.GetHouse();
Console.WriteLine(house2);
Console.ReadKey();
}
}

运行结果:

2.小结

上边例子的类图

建造者模式的使用场景:

  主要解决在软件系统中,有时候面临着"一个复杂对象"的创建工作,其通常由各个部分的子对象用一定的算法构成;由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化,但是将它们组合在一起的算法却相对稳定,这时适合用建造者模式。StringBuilder和Quartz中的JobBuilder,TriggerBuilder都使用了建造者模式。如果每添加一个部件都返回建造对象的索引,那么我们就可以实现基本的链式构造,如JobBuilder和TriggerBuilder都可以通过Fluent interface形式进行构造。

建造者模式的优点:

  1.实例的构建具体到组件层次,方便控制细节;

  2.具体的建造者相互独立,易于扩展。

建造者模式的缺点:

  当组件的组合种类很多时,需要创建很多的具体建造者类。

  

C#设计模式(5)——建造者模式的更多相关文章

  1. Java设计模式之建造者模式(Builder)

    前言: 最近一直在学习okHttp,也对其做了一些整理,okHttp和Retrofit结合大大加速我们的开发效率,源码里面采用了很多设计模式,今天我们来学习一下其中的设计模式之一建造者模式. 建造者模 ...

  2. C#设计模式(5)——建造者模式(Builder Pattern)

    一.引言 在软件系统中,有时需要创建一个复杂对象,并且这个复杂对象由其各部分子对象通过一定的步骤组合而成.例如一个采购系统中,如果需要采购员去采购一批电脑时,在这个实际需求中,电脑就是一个复杂的对象, ...

  3. 【GOF23设计模式】建造者模式

    来源:http://www.bjsxt.com/ 一.[GOF23设计模式]建造者模式详解类图关系 建造飞船 package com.test.Builder; public class AirShi ...

  4. C++设计模式之建造者模式(三)

    4.引入钩子方法的建造者模式 建造者模式除了逐步构建一个复杂产品对象外.还能够通过Director类来更加精细地控制产品的创建过程.比如添加一类称之为钩子方法(HookMethod)的特殊方法来控制是 ...

  5. 乐在其中设计模式(C#) - 建造者模式(Builder Pattern)

    原文:乐在其中设计模式(C#) - 建造者模式(Builder Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 建造者模式(Builder Pattern) 作者:webabc ...

  6. 折腾Java设计模式之建造者模式

    博文原址:折腾Java设计模式之建造者模式 建造者模式 Separate the construction of a complex object from its representation, a ...

  7. C#设计模式之四建造者模式(Builder Pattern)【创建型】

    一.引言 今天我们要讲讲Builder模式,也就是建造者模式,当然也有叫生成器模式的,英文名称是Builder Pattern.在现实生活中,我们经常会遇到一些构成比较复杂的物品,比如:电脑,它就是一 ...

  8. Java 设计模式之建造者模式(四)

    原文地址:Java 设计模式之建造者模式(四) 博客地址:http://www.extlight.com 一.前言 今天继续介绍 Java 设计模式中的创建型模式--建造者模式.上篇设计模式的主题为 ...

  9. GOF23设计模式之建造者模式

    GOF23设计模式之建造者模式 场景: 我们要建造一个复杂的产品.比如:神州飞船,Iphone.这个复杂的产品的创建.有这样的一个问题需要处理: 装配这些子组件是不是有个步骤问题? 实际开发中,我们所 ...

  10. java设计模式3——建造者模式

    java设计模式3--建造者模式 1.建造者模式介绍: 建造者模式属于创建型模式,他提供了一种创建对象得最佳方式 定义: 将一个复杂对象的构建和与它的表示分离,使得同样的构建过程可以创建不同的表示 主 ...

随机推荐

  1. Json中Date映射到model

    @DateTimeFormat(pattern="yyyy-MM-dd") private Date nenddate; public Date getNenddate() { r ...

  2. Codeforces340 E. Iahub and Permutations

    Codeforces题号:#340E 出处: Codeforces 主要算法:思维+DP 难度:4.8 题意: 有一个长度为$n$的排列(即各元素互不相同),其中有一些为-1.现要求将数填到这些-1上 ...

  3. 允许外网访问MySQL

    1:设置mysql的配置文件     /etc/mysql/my.cnf     找到 bind-address  =127.0.0.1  将其注释掉://作用是使得不再只允许本地访问:  重启mys ...

  4. AtCoder 瞎做

    目录 ARC 058 E - 和風いろはちゃん / Iroha and Haiku 题意 题解 技巧 代码 ARC 059 F - バイナリハック / Unhappy Hacking 题意 题解 技巧 ...

  5. Codeforces Round #453 (Div. 1) D. Weighting a Tree(构造)

    题意 一个 \(n\) 个点 \(m\) 条边的无向连通图中每个点都有一个权值,现在要求给每条边定一个权值,满足每个点的权值等于所有相连的边权之和,权值可负. 题解 如果图是一棵树,那么方案就是唯一的 ...

  6. redis主从复制和sentinel配置高可用

    一:redis主从配置1.环境准备 master : 192.168.50.10 6179 slave1: 192.168.50.10 6279 slave2: 192.168.50.10 63792 ...

  7. Linux网络基础-总

    目录 Linux网络基础 一.网卡和数据包的转发 1.收包流程 二.多网卡bonding 三.SR-IOV 四.DPDK 五.TUN/TAP 六.Linux bridge 和VLAN 七.TCP/IP ...

  8. 「SCOI2015」小凸玩密室 解题报告

    「SCOI2015」小凸玩密室 虽然有心里在想一些奇奇怪怪的事情的原因,不过还是写太久了.. 不过这个题本身也挺厉害的 注意第一个被点亮的是任意选的,我最开始压根没注意到 \(dp_{i,j}\)代表 ...

  9. nginx proxy文件编写总结

    upstream.conf upstream api { server 192.168.10.10:8080; server 192.168.10.20:8080;} server{ listen 4 ...

  10. uvaLive6837 ThereIsNoAlternative (kruskal)

    题意:给一个联通图,求出不可替代的边,即存在于所有最小生成树中的边,的数量和它们边权之和 首先kruskal跑出一个最小生成树,枚举其中所有的边,若把这条边去掉以后再跑kruskal答案不是最小,则这 ...