看了好几处关于builder模式的书和博客,总感觉不是很清楚,感觉不少书上的说的也不是很准确。最后还是看回圣经《设计模式》。看了好久终于感觉明白了一点了。

意图:

builder模式提出的目的就是为了解决将一个复杂对象的构建和它的表示分离,使得同样的构建过程可以创建不的表示的问题。

我们来看一个例子:

假设我们要画一些人,例如游戏里面的各类人。一般情况下,人有头,手,腿。但是在游戏里面,人可能没有手,也可能不止两只手。即所谓的三头六臂。如下:

那么,在这种情况下应该怎么去生产这些人?

一开始可能会用这个思路:抽象出一个人的基类,然后不同的人继承自这个基类得到一系列的子类。

这个方法有很多缺点:

1、首先很明显的一点是:不同类型的人差别极大,因为人不是都相同的手相同的腿等,以至于很难抽象出共同点。所以抽象出一个基类本身就不现实。

2、利用这种通过继承产生不同人的方法会造成子类的泛滥,比如每多一只手就要多一个子类,造成代码的可拓展性问题,管理问题等。

那么就要用到下面说的builder模式。

builder模式的适用性:

 在以下情况使用builder模式
  • 当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式时。
  • 当构造过程必须允许被构造的对象有不同的表示时。

我们注意,builder模式有两个要注意的点,一个是创建复杂对象的算法独立于对象的组成部分以及它们的装配方式。一个是被构造的对象有不同的表示。但我们不能被“复杂对象”所误导,并不是说将复杂对象进行封装就是builder模式了。

我们先来看看模式的结构图:

其中各部分的作用为:

• Builder:
      为创建一个Product对象的各个部件指定抽象接口。
• ConcreteBuilder
  实现Builder的接口以构造和装配该产品的各个部件。
  定义并明确它所创建的表示。
  提供一个检索产品的接口。
• Director
  构造一个使用Builder接口的对象。
•Product
  表示被构造的复杂对象。ConcreteBuilder 创建该产品的内部表示并定义它的装配过程。
  包含定义组成部件的类,包括将这些部件装配成最终产品的接口。

也就是说,导向器决定构造一个product的算法,比如构造一个人,导向器就决定他的构造顺序:

1、addHead();   2、addArm();    3、addArm();    4、addArm();   5、addLeg();    6、addLeg();

然后导向器通过调用builder的接口来实现具体构造。这样就实现了创建复杂对象的算法和对象的组成部分以及它们的装配方式的分离。

也就是说,builder模式要得到Product并不是通过concreteBuilder一步得到的,它是通过在导向器Director所列出的方法顺序下一步步进行构造得到。并在最后一步后得到一个最终的产品。

我们来看看Director和ConcreteBuilder的协作过程:

其中getResult()方法是得到构造好的实例对象。

使用builder模式的效果如下:

1、Builder对象提供给导向器的抽象接口的具体实现是在Builder内部,那么在改变接口内部的具体实现的时候就不会对导向器使用接口产生影响。它隐藏了产品是如何装配的。另外,因为Builder的抽象接口是一定的,那么你在改变产品的内部结构的时候只需要定义一个新的生成器CuncreteBuilder。然后使用这个生成器去装配产品即可。

2、它将构造代码和表示代码分开,builder模式通过封装一个复杂对象的创建和表示方式提高了对象的模块性。客户不需要知道定义产品内部的结构的类的所有信息。这些类是不出现在builder接口中的。每个ConcerteBuilder包含了创建和装配一个特定产品的所有代码。这些代码只需要写一次,然后不同的Director可以复用它以在相同的部件集合的基础上构作不同的Product。

3、它使你可对构造过程进行更精细的控制。Builder模式与一下子就生成产品的创建型模式不同,他是在Director的控制下一步步构造产品。仅当产品构造完成后才从生成器取回它。因此Builder接口相比其他创建型模式能更好地反应产品构造过程。这使你能更精细地控制构建过程,从而能更精细地控制所得产品的内部结构。

那么根据上述画出开始时的问题的builder模式的解决方案的结构图如下:

根据图就不难写出代码了。  

值得注意的是,1、产品Product没有抽象类 通常情况下,由具体生成器生成的产品,它们的表示相差很大,以至于给不同的产品以公共父类没有什么。

2、在Builder中缺省的方法为空。在C++中,生成方法故意不声明为纯虚函数,而是把它们定义为空方法,这使得客户可以只重定义他们所感兴趣的操作。

c++设计模式系列----builder模式的更多相关文章

  1. 设计模式:Builder模式

    设计模式:Builder模式 一.前言    今天我们讨论一下Builder建造者模式,这个Builder,其实和模板模式非常的像,但是也有区别,那就是在模板模式中父类对子类中的实现进行操作,在父类之 ...

  2. Java设计模式之builder模式

    Java设计模式之builder模式 今天学mybatis的时候,知道了SQLSessionFactory使用的是builder模式来生成的.再次整理一下什么是builder模式以及应用场景. 1. ...

  3. Java设计模式-建造者(Builder)模式

    目录 由来 使用 1. 定义抽象 Builder 2. 定义具体 Builder类 3. 定义具体 Director类 4. 测试 定义 文字定义 结构图 优点 举例 @ 最近在看Mybatis的源码 ...

  4. [转]C++设计模式:Builder模式

    Builder模式要解决的问题是,当我们要创建很复杂的对象时,有时候需要将复杂对象的创建过程和这个对象的表示分离开来.由于在每一步的构造过程中可以映入不同参数,所以步骤相同但是最后的对象却不一样.也就 ...

  5. 设计模式之Builder模式

    一.感性认识 二.Builder模式 1.定义 一个复杂对象的构建与其表示相分离,使得同样的构建过程可以创建不同的表示.即构建过程相同,但是子部件却不相同. 2.结构说明 Builder: 创建者接口 ...

  6. Java设计模式--Java Builder模式

    1.Java Builder模式主要是用一个内部类去实例化一个对象,避免一个类出现过多构造函数,而且构造函数如果出现默认参数的话,很容易出错. public Person(String name) P ...

  7. PHP设计模式系列 - 外观模式

    外观模式 通过在必需的逻辑和方法的集合前创建简单的外观接口,外观设计模式隐藏了调用对象的复杂性. 外观设计模式和建造者模式非常相似,建造者模式一般是简化对象的调用的复杂性,外观模式一般是简化含有很多逻 ...

  8. 《Android源码设计模式》--Builder模式

    No1: 将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示 No2: 在Android源码中,最常用到的Builder模式就是AlertDialog.Builder No3: ...

  9. akka设计模式系列-While模式

    While模式严格来说是while循环在Akka中的合理实现.while是开发过程中经常用到的语句之一,也是绝大部分编程语言都支持的语法.但while语句是一个循环,如果循环条件没有达到会一直执行wh ...

随机推荐

  1. BZOJ4850/BZOJ2216 JSOI2016灯塔/Poi2011Lightning Conductor(决策单调性)

    即对每个i最大化hj-hi+sqrt(|i-j|).先把绝对值去掉,正反各做一次即可.注意到当x>y时,sqrt(x+1)-sqrt(x)<sqrt(y+1)-sqrt(y),所以若对于i ...

  2. CF724E Goods transportation 最小割 DP

    照惯例CF的题不放原题链接... 题意:一个序列上有n个点,每个点有权值pi和si.表示这个点一开始有pi个物品,最多可以卖出si个物品,每个点都可以把物品向编号更大的点运输,但是对于i < j ...

  3. POJ1742 Coins(男人八题之一)

    前言 大名鼎鼎的男人八题,终于见识了... 题面 http://poj.org/problem?id=1742 分析 § 1 多重背包 这很显然是一个完全背包问题,考虑转移方程: DP[i][j]表示 ...

  4. 高效率JavaScript代码的编写技巧

    使用DocumentFragment优化多次append 添加多个dom元素时,先将元素append到DocumentFragment中,最后统一将DocumentFragment添加到页面.该做法可 ...

  5. js正则:两边字符固定,中间任意字符

    求些一个js正则!两边字符固定,中间任意字符.在一个长字符串里面匹配一小段,这一小段字符串开头和结尾都是固定的字符,就是中间是任意长度的字符.怎么写? /aa.+aa/ aa是你的固定字符,如果是反斜 ...

  6. selenium - webdriver - 设置元素等待

    隐式等待:implicitly_wait(value), value默认是0 from selenium import webdriverfrom selenium.common.exceptions ...

  7. Mobile phones POJ - 1195 二维树状数组求和

    Suppose that the fourth generation mobile phone base stations in the Tampere area operate as follows ...

  8. mysql的时间函数整理

      转:这里总结的非常齐全: http://fengbin2005.iteye.com/blog/1999763   Mysql时间函数 对于每个类型拥有的值范围以及并且指定日期何时间值的有效格式的描 ...

  9. saltstack:multi-master configuration

    官方手册地址:http://docs.saltstack.com/topics/tutorials/multimaster.html 总结起来,有以下几步: Create a redundant ma ...

  10. Getting Private/Public IP address of EC2 instance using AWS-cli [closed]

    For private IP address: aws ec2 describe-instances --instance-ids i-b78a096f | grep PrivateIpAddress ...