C++设计模式12--建造者模式----创建复杂对象的工程师

简介:

建造者(Builder)模式

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

 

对象性质的建造

 

有些情况下,一个对象会有一些重要的性质,在它们没有恰当的值之前,对象不能作为一个完整的产品使用。比如,一个电子邮件有发件人地址、收件人地址、主题、内容、附录等部分,而在最起码的收件人地址未被赋值之前,这个电子邮件不能发出。

 

有些情况下,一个对象的一些性质必须按照某个顺序赋值才有意义。在某个性质没有赋值之前,另一个性质则无法赋值。这些情况使得性质本身的建造涉及到复杂的商业逻辑。

 

这时候,此对象相当于一个有待建造的产品,而对象的这些性质相当于产品的零件,建造产品的过程就是组合零件的过程。由于组合零件的过程很复杂,因此,这些"零件"的组合过程往往被"外部化"到一个称作建造者的对象里,建造者返还给客户端的是一个全部零件都建造完毕的产品对象。

 

 

 

Builder模式的结构:

 

①建造者(Builder)角色:给出一个抽象接口,以规范产品对象的各个组成成分的建造。一般而言,此接口独立于应用程序的商业逻辑。模式中直接创建产品对象的是具体建造者(ConcreteBuilder)角色。具体建造者类必须实现这个接口所要求的方法:一个是建造方法,另一个是结果返还方法。

 

②具体建造者(Concrete Builder)角色:担任这个角色的是于应用程序紧密相关的类,它们在应用程序调用下创建产品实例。这个角色主要完成的任务包括:

 

③实现Builder角色提供的接口,一步一步完成创建产品实例的过程。

在建造过程完成后,提供产品的实例。

指导者(Director)角色:担任这个角色的类调用具体建造者角色以创建产品对象。导演者并没有产品类的具体知识,真正拥有产品类的具体知识的是具体建造者对象。

 

④产品(Product)角色:产品便是建造中的复杂对象。

 

指导者角色是于客户端打交道的角色。导演者角色将客户端创建产品的请求划分为对各个零件的建造请求,再将这些请求委派给具体建造者角色。具体建造者角色是做具体建造工作的,但却不为客户端所知。

 

 

程序举例:

该程序演示了Builder模式一步一步完成构件复杂产品的过程。用户可以控制生成过程以及生成不同对象。

《大话设计模式》举了一个很好的例子——建造小人,一共需建造6个部分,头部、身体、左右手、左右脚。

与工厂模式不同,建造者模式是在导向者的控制下一步一步构造产品的。建造小人就是在控制下一步步构造出来的。创建者模式可以能更精细的控制构建过程,从而能更精细的控制所得产品的内部结构。

我们要生成不同的小人

  1. // 产品[product]角色  
  2. class Person    // 待建造的l人  
  3. {  
  4. public :  
  5.     void SetHead(std::string head)  
  6.     {  
  7.         this->m_head = head;  
  8.     }  
  9.   
  10.     std::string GetHead( )  
  11.     {  
  12.         return this->m_head;  
  13.     }  
  14.   
  15.     void SetBody(std::string body)  
  16.     {  
  17.         this->m_body = body;  
  18.     }  
  19.   
  20.     std::string GetBody( )  
  21.     {  
  22.         return this->m_body;  
  23.     }  
  24.   
  25.     void SetFoot(std::string foot)  
  26.     {  
  27.         this->m_foot = foot;  
  28.     }  
  29.   
  30.     std::string GetFoot( )  
  31.     {  
  32.         return this->m_foot;  
  33.     }  
  34.   
  35.     void ShowPerson( )         // 显示当前的人的信息  
  36.     {  
  37.         std::cout <<"This is a person has" <<std::endl;  
  38.         std::cout <<"HEAD : "<<this->m_head <<", BODY : " <<this->m_body <<", FOOT : " <<this->m_foot <<std::endl;  
  39.     }  
  40.   
  41. protected :  
  42.     std::string m_head;     // 人的头  
  43.     std::string m_body;     // 人的躯体  
  44.     std::string m_foot;     // 人的脚  
  45. };  


不同的小人由不同的建造工程师进行设计

  1. // 建造者[build]角色  
  2. class Builder    // 建造者的基类  
  3. {  
  4. public :  
  5.     Builder()             // 无参数的构造函数  
  6.     {  
  7.         this->m_person = new Person( );  
  8.     }  
  9.   
  10.     virtual ~Builder( )      // 析构函数  
  11.     {  
  12.         delete this->m_person;  
  13.     }  
  14.   
  15.   
  16.     Person*& GetPerson( )            // 获取建造者建造的人对象  
  17.     {  
  18.         return this->m_person;  
  19.     }  
  20.   
  21.     virtual void BuildHead( ) = 0;          // 建造人的头  
  22.   
  23.     virtual void BuildBody( ) = 0;          // 建造人的躯体  
  24.   
  25.     virtual void BuildFoot( ) = 0;          // 建造人的脚  
  26.   
  27. protected :  
  28.     Person *m_person;  
  29. };  
  30.   
  31. // 胖子的实际建造者-=> 具体建造者[Concrete Builder]  
  32. class FatBuilder : public Builder  
  33. {  
  34. public :  
  35.     FatBuilder()             // 无参数的构造函数  
  36.     :Builder( )  
  37.     {  
  38.        // this->m_person = new Person( );  
  39.     }  
  40.   
  41.     virtual ~FatBuilder( )      // 析构函数  
  42.     {  
  43.     }  
  44.   
  45.     void BuildHead( )          // 建造人的头  
  46.     {  
  47.         this->m_person->SetHead("Fat Head");  
  48.         std::cout <<"Build Fat head..." <<std::endl;  
  49.     }  
  50.   
  51.     void BuildBody( )         // 建造人的躯体  
  52.     {  
  53.         this->m_person->SetBody("Fat Body");  
  54.         std::cout <<"Build Fat body..." <<std::endl;  
  55.     }  
  56.   
  57.     void BuildFoot( )          // 建造人的脚  
  58.     {  
  59.         this->m_person->SetFoot("Fat Foot");  
  60.         std::cout <<"Build Fat foot..." <<std::endl;  
  61.     }  
  62.   
  63. };  
  64.   
  65.   
  66.   
  67. // 瘦人的实际建造者-=>具体建造者[Concrete Builder]  
  68. class ThinBuilder : public Builder  
  69. {  
  70. public :  
  71.     ThinBuilder( )             // 无参数的构造函数  
  72.     :Builder( )  
  73.     {  
  74.        // this->m_person = new Person( );  
  75.     }  
  76.   
  77.     virtual ~ThinBuilder( )      // 析构函数  
  78.     {  
  79.     }  
  80.       
  81.     void BuildHead( )          // 建造人的头  
  82.     {  
  83.         this->m_person->SetHead("Thin Head");  
  84.         std::cout <<"Build thin head..." <<std::endl;  
  85.     }  
  86.   
  87.     void BuildBody( )         // 建造人的躯体  
  88.     {  
  89.         this->m_person->SetBody("Thin Body");  
  90.         std::cout <<"Build thin body..." <<std::endl;  
  91.     }  
  92.   
  93.     void BuildFoot( )          // 建造人的脚  
  94.     {  
  95.         this->m_person->SetFoot("Thin Foot");  
  96.         std::cout <<"Build thin foot..." <<std::endl;  
  97.     }  
  98.   
  99. };  


我们有一个向导来控制小人的创建,只要告诉它我们需要什么样的小人,向导会自动为我们设计

  1. // 向导  
  2. class Director  
  3. {  
  4. public :  
  5.     Director(Builder *builder)          // 向导类构造函数  
  6.     {  
  7.         this->m_builder = builder;  
  8.     }  
  9.   
  10.     void CreatePerson( )  
  11.     {     
  12.         this->m_builder->BuildHead( );      // 利用建造者建造人的头  
  13.         this->m_builder->BuildBody( );      // 利用建造者建造人的躯体  
  14.         this->m_builder->BuildFoot( );      // 利用建造者建造人的脚  
  15.     }  
  16.   
  17. protected:  
  18.     Builder *m_builder;         //  
  19. };  


下面看看客户端调用的代码

  1. int main()  
  2. {  
  3.     std::cout <<"Start create thin person..." <<std::endl;  
  4.     ThinBuilder thinBuilder;  
  5.     Director thinDirector(&thinBuilder);  
  6.     thinDirector.CreatePerson( );  
  7.     Person *thinPerson = thinBuilder.GetPerson( );  
  8.     thinPerson->ShowPerson( );  
  9.     std::cout <<std::endl;  
  10.   
  11.   
  12.     std::cout <<"Start create fat person..." <<std::endl;  
  13.     FatBuilder fatBuilder;  
  14.     Director fatDirector(&fatBuilder);  
  15.     fatDirector.CreatePerson( );  
  16.     Person *fatPerson = fatBuilder.GetPerson( );  
  17.     fatPerson->ShowPerson( );  
  18.     std::cout <<std::endl;  
  19.   
  20.   
  21.     return 0;  
  22. }  


转载:http://blog.csdn.net/gatieme/article/details/21512591

目录
相关文章
|
24天前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
10天前
|
设计模式 存储 Java
【再谈设计模式】备忘录模式~对象状态的守护者
备忘录模式属于行为型设计模式。它的主要目的是在不破坏对象封装性的前提下,捕获并外部化一个对象的内部状态,以便之后可以将该对象恢复到这个状态。原发器(Originator):创建一个备忘录,用于记录当前时刻它的内部状态。原发器还可以使用备忘录来恢复其内部状态。备忘录(Memento):存储原发器对象的内部状态。备忘录应该防止原发器以外的其他对象访问其内部状态。负责人(Caretaker):负责保存备忘录,但不能对备忘录的内容进行操作或检查。
181 82
|
1月前
|
设计模式 供应链 安全
【再谈设计模式】中介者模式 - 协调对象间交互的枢纽
中介者模式定义了一个中介对象来封装一组对象之间的交互方式。中介者使得各对象之间不需要显式地相互引用,从而降低了它们之间的耦合度。它通过将对象之间的交互逻辑集中到中介者对象中,使得系统的结构更加清晰,易于维护和扩展。
53 18
【再谈设计模式】中介者模式 - 协调对象间交互的枢纽
|
27天前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
1月前
|
设计模式 Java Go
【再谈设计模式】状态模式~对象行为的状态驱动者
状态模式属于行为型设计模式。它将对象的行为封装在不同的状态类中,使得对象在不同的状态下表现出不同的行为。上下文(Context):这是一个包含状态对象的类,它定义了客户感兴趣的接口,并维护一个具体状态对象的引用。上下文将操作委托给当前的状态对象来处理。抽象状态(State):这是一个抽象类或者接口,它定义了一个特定状态下的行为接口。所有具体的状态类都实现这个接口。具体状态(Concrete State):这些是实现抽象状态接口的类,每个具体状态类实现了与该状态相关的行为。
73 18
|
17天前
|
设计模式 算法 Java
【再谈设计模式】访问者模式~操作对象结构的新视角
  访问者模式是一种行为设计模式,旨在解决对象结构与操作逻辑的耦合问题。在软件系统开发中,当面临复杂的对象结构(如多种类型对象组成的树形或图形结构),且需要对这些对象执行不同操作时,传统方式将操作直接写在对象类中会导致类职责过多,不利于维护和扩展。而访问者模式通过将操作与对象结构分离,允许在不改变现有对象结构的情况下定义新操作,元素接受访问者访问,访问者定义对不同类型元素的操作逻辑,从而为应对这种复杂情况提供了有效的解决方案。
26 0
|
24天前
|
安全 编译器 C语言
【C++篇】深度解析类与对象(中)
在上一篇博客中,我们学习了C++类与对象的基础内容。这一次,我们将深入探讨C++类的关键特性,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载、以及取地址运算符的重载。这些内容是理解面向对象编程的关键,也帮助我们更好地掌握C++内存管理的细节和编码的高级技巧。
|
24天前
|
存储 程序员 C语言
【C++篇】深度解析类与对象(上)
在C++中,类和对象是面向对象编程的基础组成部分。通过类,程序员可以对现实世界的实体进行模拟和抽象。类的基本概念包括成员变量、成员函数、访问控制等。本篇博客将介绍C++类与对象的基础知识,为后续学习打下良好的基础。
|
2月前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
82 19
|
2月前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
78 13