意图

将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以表示不同的表示。

适用性

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

UML图

  • Builder ——为创建一个Product对象的各个部件指定抽象接口。
  • ConcreteBuilder ——实现Builder的接口以构造和装配该产品的各个部件。

    ——定义并明确它所创建的表示。

    ——提供一个检索产品的接口。

  • Director ——构造一个使用Builder接口的对象。
  • Product ——表示被构造的复杂对象。
  • ConcreteBuilder创建该产品的内部表示并定义它的装配过程。

    ——包含定义组成部件的类,包括将这些部件装配成最终产品的接口。

代码示例

  ConcreteBuilder中需要提供构建产品的的各个部件的函数,以及对外获取被创建产品的接口。还是以创建迷宫为例,迷宫产品组件包括迷宫自身、房间、门,这些组件创建函数(BuildPart)需定义为虚函数,以便子类继承实现。在产品各部分创建好了后,需要通过对外的接口GetResult来获取创建的产品,本例GetMaze返回创建的迷宫。

  1. class MazeBuilder{
  2. public:
  3. virtual void BuildMaze(){}
  4. virtual void BuildRoom(int Room){}
  5. virtual void BuildDoor(int roomFrom, int RoomTo){}
  6.  
  7. virtual Maze* GetMaze(){return 0;}
  8. protected:
  9. MazeBuilder();
  10. };

  因为builder中BuildPart和GetResult函数都是虚函数,所以Director并不关心继承Builder的子类实现,Director只需要按部就班的调用Builder的BuildPart函数来创建产品,如下例:

builder.BuildMaze();//创建迷宫

builder.BuildRoom(1);//创建1号房间

builder.BuildRoom(2);//创建2号房间

builder.BuildDoor(1,2);//创建1号房间和2号房间的门

在产品各个部件创建完成后,调用:

build.GetMaze();//获取已创建迷宫

  1. Maze* MazeGame::CreateMaze(MazeBuilder& builder){
  2. builder.BuildMaze();
  3.  
  4. builder.BuildRoom(1);
  5. builder.BuildRoom(2);
  6. builder.BuildDoor(1,2);
  7.  
  8. return builder.GetMaze();
  9. }
  1. MazeBuilder基类提供实现产品的接口,具体实现还需要子类来实现。这里StandardMazeBuilder继承MazeBuilder并实现了基本功能。
  1. StandardMazeBuilder类声明:
  1. class StandardMazeBuilder:public MazeBuilder{
  2. public:
  3. StandardMazeBuilder();
  4.  
  5. virtual void BuildMaze();
  6. virtual void BuildRoom(int);
  7. virtual void BuildDoor(int,int);
  8.  
  9. virtual Maze* GetMaze();
  10. private:
  11. Direction CommonWall(Room *, Room *);
  12. Maze * _currentMaze;
  13. }

StandardMazeBuilder类实现:

  1. #include "Builder.h"
  2.  
  3. StandardMazeBuilder::StandardMazeBuilder(){
  4. _currentMaze = 0;
  5. }
  6.  
  7. void StandardMazeBuilder::BuildMaze(){
  8. _currentMaze = new Maze;
  9. }
  10.  
  11. Maze* StandardMazeBuilder::GetMaze(){
  12. return _currentMaze;
  13. }
  14.  
  15. void StandardMazeBuilder::BuildRoom(int n){
  16. if(!_currentMaze->RoomNo(n)){
  17. Room* room = new Room(n);
  18. _currentMaze->AddRoom(room);
  19.  
  20. room->SetSide(North, new Wall);
  21. room->SetSide(South, new Wall);
  22. room->SetSide(East, new Wall);
  23. room->SetSide(West, new Wall);
  24. }
  25. }
  26.  
  27. void StandardMazeBuilder::BuildDoor(int n1, int n2){
  28. Room *r1 = _currentMaze->RoomNo(n1);
  29. Room *r2 = _currentMaze->RoomNo(n2);
  30. Door *d = new Door(r1,r2);
  31.  
  32. r1->SetSide(CommonWall(r1,r2),d);
  33. r2->SetSide(CommonWall(r2,r1),d);
  34. }

   在实现了Director和Builder后,Client无需关心具体产品的具体实现,只需要新建该Builder类,传递给Director,就能创建出不同的产品了。

  1. #include "Direcotr.h"
  2. #include "StandardMazeBuilder.h"
  3.  
  4. int Main(){
  5.  
  6. Maze *maze;
  7. MazeGame game;
  8. StandardMazeBuilder builder;
  9.  
  10. game.CreateMaze(builder);
  11. maze = builder.GetMaze();
  12. return 0;
  13. }

 

优缺点

优点:

  • 将一个复杂对象的创建过程封装起来。
  • 允许对象通过多个步骤来创建,并且可以改变过程(这和只有一个步骤的工厂模式不同)。
  • 向客户隐藏产品内部的实现。
  • 产品的实现可以被替换,因为客户只看到一个抽象的接口。

缺点:

  • 与工厂模式相比,采用生成器创建对象的模式,需要具备更多的领域知识。

参考文献:

GOF《设计模式》

《Head First设计模式》

设计模式之生成器(Builder)模式的更多相关文章

  1. 设计模式 笔记 生成器(建造者)模式 Builder

    //---------------------------15/04/08---------------------------- //builder 生成器(建造者)模式---对象创建型模式 /* ...

  2. Builder模式在Java中的应用

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

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

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

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

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

  5. 每天一个设计模式-7 生成器模式(Builder)

    每天一个设计模式-7 生成器模式(Builder) 一.实际问题 在讨论工厂方法模式的时候,提到了一个导出数据的应用框架,但是并没有涉及到导出数据的具体实现,这次通过生成器模式来简单实现导出成文本,X ...

  6. 设计模式03: Builder 生成器模式(创建型模式)

    Builder生成器模式(创建型模式) Builder模式缘起假设创建游戏中的一个房屋House设施,该房屋的构建由几个部分组成,且各个部分富于变化.如果使用最直观的设计方法,每个房屋部分的变化,都将 ...

  7. 设计模式十: 生成器模式(Builder Pattern)

    简介 生成器模式属于创建型模式的一种, 又叫建造者模式. 生成器模式涉及4个关键角色:产品(Product),抽象生成器(builder),具体生成器(ConcreteBuilder),指挥者(Dir ...

  8. 面向对象设计模式纵横谈:Builder 生成器模式(笔记记录)

    Builder模式的缘起 假设创建游戏中的一个房屋House设施,该房屋的构建由几个部分组成,且各个部分要富于变化. 如果使用最直观的设计方法,每一个房屋部分的变化,都将导致房屋构建的重新修正…… 动 ...

  9. 设计模式之构建者模式(Builder):初步理解

    构建者(Builder)设计模式(又叫生成器设计模式): 当一个类的内部数据过于复杂的时候(通常是负责持有数据的类,比如Config.VO.PO.Entity...),要创建的话可能就需要了解这个类的 ...

随机推荐

  1. cronatab周期性任务

    要执行周期性任务,要保证服务运行.服务名为crond:service crond start ; systemctl start crond; 配置文件 /etc/crontab cron的日志文件 ...

  2. 【前端】JavaScript中prototype和__proto__的区别

    转载请注明出处:http://www.cnblogs.com/shamoyuu/p/prototype.html 经常有小伙伴问我关于prototype和__proto__的问题,觉得有必要写一篇博客 ...

  3. spring中自定义Event事件的使用和浅析

    在我目前接触的项目中,用到了许多spring相关的技术,框架层面的spring.spring mvc就不说了,细节上的功能也用了不少,如schedule定时任务.Filter过滤器. intercep ...

  4. R语言︱文本挖掘——词云wordcloud2包

    每每以为攀得众山小,可.每每又切实来到起点,大牛们,缓缓脚步来俺笔记葩分享一下吧,please~ --------------------------- 笔者看到微信公众号探数寻理中提到郎大为Chif ...

  5. 【mysql】mysql主从复制

    mysql主从复制配置 主服务器:192.168.0.100 从服务器 192.168.0.101 主服务器配置 my.ini(window下 linux 下是my.cnf) #开启二进制日志 log ...

  6. Caused by: java.lang.ClassNotFoundException: javax.persistence.Entity

    1.错误描述 usage: java org.apache.catalina.startup.Catalina [ -config {pathname} ] [ -nonaming ] { -help ...

  7. Linux显示邮件状态等信息

    Linux显示邮件状态等信息 youhaidong@youhaidong-ThinkPad-Edge-E545:~$ finger -l Login: youhaidong Name: youhaid ...

  8. 使用Gulp进行代码压缩的步骤以及配置

    一.安装步骤 1.首先确定是否安装了node.js,如果未安装,请先安装node.js: 2.确定是否安装了包管理工具npm,如未安装请安装:npm install npm -g: 3.安装gulp: ...

  9. html->html5->css->javascript(js)->jQuery->AJAX->JSON

    首先来讲HTML,html是很多人接触到的第一种编写网页的语言,小编是在大学本科期间的编程课程上接触到html的,小小的几行代码就能编制出精彩绝伦的网页,不禁令人赞叹!HTML的语法自然不用多说,网上 ...

  10. luogu2402 奶牛隐藏

    题目描述 在一个农场里有n块田地.某天下午,有一群牛在田地里吃草,他们分散在农场的诸多田地上,农场由m条无向的路连接,每条路有不同的长度. 突然,天降大雨,奶牛们非常混乱,想要快点去躲雨.已知每个田地 ...