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

目录
相关文章
|
5月前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
513 0
|
10月前
|
设计模式 Java Apache
【设计模式】【创建型模式】建造者模式(Builder)
一、入门 什么是建造者模式? 建造者模式(Builder Pattern)是一种创建型设计模式,用于逐步构建复杂对象。 它通过将对象的构建过程与表示分离,使得相同的构建过程可以创建不同的表示。 为什么
521 14
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
设计模式 存储 Java
【再谈设计模式】备忘录模式~对象状态的守护者
备忘录模式属于行为型设计模式。它的主要目的是在不破坏对象封装性的前提下,捕获并外部化一个对象的内部状态,以便之后可以将该对象恢复到这个状态。原发器(Originator):创建一个备忘录,用于记录当前时刻它的内部状态。原发器还可以使用备忘录来恢复其内部状态。备忘录(Memento):存储原发器对象的内部状态。备忘录应该防止原发器以外的其他对象访问其内部状态。负责人(Caretaker):负责保存备忘录,但不能对备忘录的内容进行操作或检查。
391 82
|
12月前
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
12月前
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
12月前
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
设计模式 算法 Java
【再谈设计模式】访问者模式~操作对象结构的新视角
  访问者模式是一种行为设计模式,旨在解决对象结构与操作逻辑的耦合问题。在软件系统开发中,当面临复杂的对象结构(如多种类型对象组成的树形或图形结构),且需要对这些对象执行不同操作时,传统方式将操作直接写在对象类中会导致类职责过多,不利于维护和扩展。而访问者模式通过将操作与对象结构分离,允许在不改变现有对象结构的情况下定义新操作,元素接受访问者访问,访问者定义对不同类型元素的操作逻辑,从而为应对这种复杂情况提供了有效的解决方案。
247 0
|
安全 编译器 C语言
【C++篇】深度解析类与对象(中)
在上一篇博客中,我们学习了C++类与对象的基础内容。这一次,我们将深入探讨C++类的关键特性,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载、以及取地址运算符的重载。这些内容是理解面向对象编程的关键,也帮助我们更好地掌握C++内存管理的细节和编码的高级技巧。
|
存储 程序员 C语言
【C++篇】深度解析类与对象(上)
在C++中,类和对象是面向对象编程的基础组成部分。通过类,程序员可以对现实世界的实体进行模拟和抽象。类的基本概念包括成员变量、成员函数、访问控制等。本篇博客将介绍C++类与对象的基础知识,为后续学习打下良好的基础。