设计模式之建造者(Builder)模式

  存在一些情况,比如,一些对象会有一些重要的属性,在这些属性没有恰当的值之前,对象不能作为一个完整的产品使用(如一个电子邮件最起码得有收件人地址);还有一些些情况,一个对象的一些属性必须按照一定的顺序赋值才有意义,在某个属性没有赋值之前,另一个属性则无法赋值。Builder模式非常适用于此种类型的情况。

  (一)什么是Builder模式

  Builder模式是一种对象创建模式,用来隐藏复合对象的创建过程。建造者模式利用一个导演者(Director)对象和具体构造者(Concrete Builder)对象一个一个地建造出所有属性(零件),从而建造出完整的产品对象。建造者模式将产品的结构和产品的零件建造过程对客户端隐藏起来,把对建造过程进行指挥的责任和具体建造者零件的责任分割开来,达到责任划分和封装的目的。

  (二)Builder模式中的角色

  抽象建造者(Builder)角色:给出一个抽象接口,以规范产品对象的各个组成部分的建造。一般此接口独立于应用程序的商业逻辑。这个接口一般包含两种方法,创建方法和返回方法。

  具体建造者(ConcreteBuilder)角色:它必须实现创造者Builder所声明的接口,在创在完成后,提供产品实例。一般来说,每有一个具体产品,就有一个相应的具体建造类。

  导演者(Director)角色:担任这个角色的类调用具体建造者角色以创建产品对象。

  产品(Product)角色:产品就是所建造的复杂对象,一个系统中会有多于一个的产品类,这些产品类并不一定有共同的接口,而完全可以是不相互关联的。导演者角色是与客户端打交道的角色,它将客户端创造产品的请求划分为多个零件的建造请求,再将这些请求委派给具体建造者角色,具体建造者角色是做具体建造工作的。

  Builder模式的结构图如下:

(三)建造者模式的具体实现

  想象一下,有这样一种业务场景:客户端向要建造一所房子 (可以是平房,也可以是公寓),房子由多个组件构成,包括地板、墙和天花板等,从程序的角度出发,能有哪些解决方案呢?

    1)客户端自己创建房子。

    2)客户端委托具体工程队建造房子

    3)客户端寻找设计师,并给设计师配备工程队建造房子 。

  (1)客户端自己建造房子。

  客户端自己创建房子的各个组件,如果建造平房就用相应的平房地板、墙、天花板

      House house = new House();
      house.setFloor("地板");
      house.setWall("墙");
      house.setCeil("天花板");

  (2)客户端委托工程队创建房子

  客户端委托工程队建造房子 ,如果要建造平房,就选择建造平房的工程队,如果建造公寓,就选择建造公寓的工程队。

        HouseBuilder builder = new PingFangBuilder();
builder.makeFloor();
builder.makeWall();
builder.makeCeil();
House pingFang = builder.getHouse();

  貌似这种方式,看起来已经比较完美,有点面向对象的意思。但仍存在缺点,封装不够,房子具体部件的创建仍然暴露在客户端(在这里可以理解为建造房子的时候,还需要需要客户端进行督工),倘若需要满足这样一种对象创建逻辑:必须按照地板、墙、天花板的顺序建造房子,在客户端创建容易出错。此时,这种方式创建对象不能满足要求,第三种方式却能满足。

  (3)客户端寻找设计师,并给设计师配备工程队建造房子

  客户端委托设计师创建房子,房子各个组件的创建顺序可以在设计师中具体指定,具体的建造仍然交给工程队,只要工程队的调度交给设计者,也可以称为管理者。这样设计者就需要一个具体的工程队,可以选择两种方式:1)每个设计师包含一个工程队属性。2)给设计师的相应方法传入一个具体 工程队。在这里我们选择第二种方式。

  以上叙述中,房子扮演者Product的角色,设计师扮演Director角色,工程队扮演Builder角色,而具体工程队(包括平房工程队和公寓工程队)扮演ConcreteBuilder角色。具体代码实现如下:

  1)Product:House

/*
* 房子
*/
public class House {
//地板
private String floor;
//墙
private String wall;
//屋顶
private String ceil;
public String getFloor() {
return floor;
}
public void setFloor(String floor) {
this.floor = floor;
}
public String getWall() {
return wall;
}
public void setWall(String wall) {
this.wall = wall;
}
public String getCeil() {
return ceil;
}
public void setCeil(String ceil) {
this.ceil = ceil;
}
}

  2)抽象建造者:HouseBuilder

public interface HouseBuilder {
//修地板
public void makeFloor();
//修墙
public void makeWall();
//修天花板
public void makeCeil(); //返回房子
public House getHouse();
}

  3)具体建造者:PingFangBuilder、ApartmentBuilder

public class PingFangBuilder implements HouseBuilder {
private House house = new House();
@Override
public void makeFloor() {
house.setFloor("平房---房顶");
} @Override
public void makeWall() {
house.setWall("平房---墙");
} @Override
public void makeCeil() {
house.setCeil("平房---天花板");
} @Override
public House getHouse() {
return house;
}
}
public class ApartmentBuilder implements HouseBuilder {
private House house = new House();
@Override
public void makeFloor() {
house.setFloor("公寓---地板");
} @Override
public void makeWall() {
house.setWall("公寓---墙");
} @Override
public void makeCeil() {
house.setCeil("公寓---天花板");
} @Override
public House getHouse() {
return house;
} }

  4)DIrector:HouseDirector

public class HouseDirector {
public House constructHouse(HouseBuilder builder) {
builder.makeFloor();
builder.makeWall();
builder.makeCeil();
return builder.getHouse();
}
}

  5)Client

HouseBuilder builder1 = new PingFangBuilder();
HouseDirector director1 = new HouseDirector();
House pingFang = director1.constructHouse(builder1); HouseBuilder builder2 = new ApartmentBuilder();
HouseDirector director2 = new HouseDirector();
House apartment = director2.constructHouse(builder2);

  (四)什么情况下使用建造模式

   需要生成的产品对象有复杂的内部结构,。

  需要生成的产品对象的属性相互依赖,即一个属性必须在一个属性被赋值后才可以被赋值。

  在对象创建过程中,会使用系统的其他一些对象,这些对象在产品创建的过程中不易得到。

  (五)抽象工厂模式和Builder模式的区别: 

  二者都是对象创建模式,在抽象工厂模式中,每调用一次工厂对象,都会返回一个完整的产品对象,客户端可能会把这些产品组装成一个更大更复杂的产品,也可能不会。建造者模式则不同,它一点一点地构造出一个复杂的产品,而这个产品的组装过程就发生在建造者角色内部,建造者模式的客户端拿到的是一个完成的产品。

  也就是说,建造者模式处于更加宏观的尺度上,而建造者模式处于更加具体的尺度上。建造者模式所创建产品的零部件可由抽象工厂创建完成。

设计模式之建造者(Builder)模式的更多相关文章

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

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

  2. 设计模式C++描述----07.建造者(Builder)模式

    一. 概述 Builder 模式要解决的问题是:当我们要创建的对象很复杂的时候(通常是由很多其他的对象组合而成),我们要要复杂对象的创建过程和这个对象的表示(展示)分离开 来,这样做的好处就是通过一步 ...

  3. 建造者(Builder)模式

    建造者模式是对象的创建模式.建造模式可以将一个产品的内部表象(internal representation)与产品的生产过程分割开来,从而可以使一个建造过程生成具有不同的内部表象的产品对象. 产品的 ...

  4. Android 建造者(Builder)模式

    关于 Builder 模式 详述:http://blog.csdn.net/jjwwmlp456/article/details/39890699 先来张图 看到 Android  中 使用了 Bui ...

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

    将一个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示                                                         --<设计模 ...

  6. 设计模式之生成器(Builder)模式

    意图 将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以表示不同的表示. 适用性 当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式时. 当构造过程必须允许被构造的对象有不同的表 ...

  7. Builder模式(建造者模式)

    在设计模式中对Builder模式的定义是用于构建复杂对象的一种模式,所构建的对象往往需要多步初始化或赋值才能完成.那么,在实际的开发过程中,我们哪些地方适合用到Builder模式呢?其中使用Build ...

  8. Builder模式在Java中的应用

    在设计模式中对Builder模式的定义是用于构建复杂对象的一种模式,所构建的对象往往需要多步初始化或赋值才能完成.那么,在实际的开发过程中,我们哪些地方适合用到Builder模式呢?其中使用Build ...

  9. Builder模式在Java中的应用(转)

    在设计模式中对Builder模式的定义是用于构建复杂对象的一种模式,所构建的对象往往需要多步初始化或赋值才能完成.那么,在实际的开发过程中,我们哪些地方适合用到Builder模式呢?其中使用Build ...

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

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

随机推荐

  1. shell脚本,当用sed删除某一文件里面的内容时,并追加到同一个文件会出现问题。

    shell脚本,当用sed删除某一文件里面的内容时,并追加到同一个文件会出现问题.因为初始文件和写入文件是一个文件这是失败的.需要追加到另一个文件,然后再用mv进行操作.[root@localhost ...

  2. Quartz监听的端口

    上海移通短信网关:556重庆移动短信网关:557消息中心后台维护服务:558网页订单数据同步服务:559基础数据同步程序:560短信数据扣除服务:565基础数据维护服务:589推送数据抓取服务:222 ...

  3. HTML5 FormData 模拟表单控件 支持异步上传二进制文件 移动端

    FormData是XMLHttpRequest Level 2添加的一个新的接口,利用FormData对象,我们可以通过JavaScript用一些键值对来模拟一系列表单控件,还可以使用XMLHttpR ...

  4. c++ 结构体,设置物品体积并输出物品属性

    #include <iostream> using namespace std; struct box { char maker[40]; float height; float widt ...

  5. 小型LAMP搭建

    一.dns的搭建 安装dns服务 yum install bind 修改dns的主配置文件 [root@234c17 named]# vim /etc/named.conf // // named.c ...

  6. log4j日志输出到文件的配置

    1.Maven的dependency 2.log4j.properties的配置 3.Junit的Test类 4.web.xml的配置(非必要) 5.spring的db.config的配置(非必要) ...

  7. LeetCode(137) Single Number II

    题目 Given an array of integers, every element appears three times except for one. Find that single on ...

  8. LeetCode(107) Binary Tree Level Order Traversal II

    题目 Given a binary tree, return the bottom-up level order traversal of its nodes' values. (ie, from l ...

  9. 《C/C++工程师综合练习卷》之小试牛刀

    第一套练习之感受 刚刚注册了牛客网的账号,准备在此衡量一下水平,好好磨练一番.看到推荐练习<C/C++工程师综合练习卷>,oh,20道题,2个小时.由于木有经验,好一番紧张~ 结果用了20 ...

  10. 菜鸟的《Linux程序设计》学习—shell script

    1. 认识shell script shell script是利用shell的功能缩写的一个"程序",这个程序是使用纯文本文件,将一些shell的语法与命令(含外部命令)写在里面, ...