设计模式-建造者模式

建造者模式

应用场景

​ 当一个类的构造函数参数有多个时,比如超过四个,而且这些参数有些是可选时,我们通常有两种方法来构建它的对象。例如我们现在有一个如下类Computer,其中CPUram是必填参数,而其他3个是可选参数,那么我们如何构造这个类的实例呢,通常有两种常用方式:

c 复制代码
#include <iostream>
#include <string>

class Computer1{
private:
    //必选
    std::string cpu;
    std::string ram;
    
    //可选
    std::string keyboard;
    std::string display;
public:
    Computer1();
    Computer1(std::string cpu,std::string ram){
        Computer1(cpu,ram,"罗技");
    }
    
    Computer1(std::string cpu,std::string ram,std::string keyboard){
        Computer1(cpu,ram,"三星显示器");
    }
    Computer1(std::string cpu,std::string ram,std::string keyboard,std::string display)
        :cpu(cpu),ram(ram),keyboard(keyboard),display(display){}

};

class Computer2{
private:
    //必选
    std::string cpu;
    std::string ram;
    
    //可选
    std::string keyboard;
    std::string display;
public:
    Computer2(std::string cpu,std::string ram):cpu(cpu),ram(ram){}
    void setCpu(std::string name){this->cpu = name;}
    void SetRam(std::string name){this->ram = name;}
    void setKeyboard(std::string name){this->keyboard = name;}
    void setDisplay(std::string name){this->display = name;}
};


 

​ 那么这两种方式有什么弊端呢?

  1. 第一种方式主要是使用及阅读不方便,参数较多,调用者分不清参数的具体含义。同时,参数太多容易混淆
  2. 第二种方式在构建过程中对象的属性状态因为类中的属性是分布设置的,容易发生变化,造成错误

建造者模式的实现要点

  • Computer中创建一个静态的Builder,负责产品Computer对象的创建
  • 创建一个指挥者Director,指挥不同的Builder对象创建不同Computer对象
  • Builder中创建一个public的构造函数,参数为Computer中必填的参数,本例子中的CPUram
  • Builder中创建设置函数,对computer中那些可选的参数进行赋值
  • Builder中创建一个build()方法,在其中构建Computer的实例并返回

建造者模式的实现方式

建造者模式由指挥者、抽象建造者、具体建造者和产品等4个要素构成,其基本结构如下。

指挥者角色(Director) :

  • 构建一个是由Builder接口创建的对象

抽象建造则模式(Builder):

  • 为创建一个Product对象的各个部件指定的抽象接口

具体建造者模式 (ConcreteBuilder):

  • 实现Builder接口,构造和装配多个部件

产品角色(Product):

  • 待生成的具体产品

设计类图

代码实现

cpp 复制代码
#include <iostream>
#include <string>

using namespace std;

class Product{


private:
        string partA;//必选参数
        string partB;
        string partC;
public:
    Product(std::string partA):partA(partA){}
    //省略get   以及toString方法
    void setPartA(string param){this->partA = param;}
    void setPartB(string param){this->partB = param;}   
    void setPartC(string param){this->partC = param;}      

    void show(){
        cout<<"Product[partA="<<partA<<",partB = "<<partB<<",partC =  "<<partC<<"]";
        
    }
};

//虚基类
class Builder{
    public:
        virtual void        setPartA() = 0;
        virtual void        setPartB() = 0;
        virtual void        setPartC() = 0;
        
        virtual Product*    build() = 0;
};

class ConcreteBuilder: public Builder{
private:
    Product * product;
public: 
    ConcreteBuilder(std::string partA){
        product = new Product(partA);
    }

    void setPartA(){product->setPartA("A");}
    void setPartB(){product->setPartB("B");}
    void setPartC(){product->setPartC("C");}
    Product* build(){return product;}
};

class Director{

private:
    Builder * builder;
public:
    Director(Builder* builder){//传入build子类对象
        this->builder = builder;
    }

    Product* constructA(){
        
        builder->setPartB();
        builder->setPartC();
        return builder->build();
    }

      Product* constructB(){
        builder->setPartA();    
        builder->setPartC();
        return builder->build();
    }
};

void BuildTest(){
    Builder * builder  = new ConcreteBuilder("partA");
    Director* director = new Director(builder);
    
    Product * productA = director->constructA();
    Product * productB = director->constructB();
    
    productA->show(),productB->show();
     
}

int main()
{
    BuildTest();
    return 0;
}

建造者模式的优点

  • 封装性: 是客户端不必知道产品内部组成的细节
  • 建造者独立,易扩展
  • 便于控制细节风险,可以对建造过程逐步细化,而不对其他模块产生任何影响

建造者模式的缺点

  • 增加类数量 : 产生多余的 Builder 对象 ;
  • 内部修改困难 : 如果 产品内部发生变化 , 建造者也要相应修改 ;
相关推荐
叶羽西7 小时前
Android Studio打开一个外部的Android app程序
android·ide·android studio
lxyzcm10 小时前
深入理解C++23的Deducing this特性(上):基础概念与语法详解
开发语言·c++·spring boot·设计模式·c++23
越甲八千10 小时前
重温设计模式--单例模式
单例模式·设计模式
Vincent(朱志强)10 小时前
设计模式详解(十二):单例模式——Singleton
android·单例模式·设计模式
诸葛悠闲12 小时前
设计模式——桥接模式
设计模式·桥接模式
odng15 小时前
IDEA自己常用的几个快捷方式(自己的习惯)
java·ide·intellij-idea
捕鲸叉16 小时前
C++软件设计模式之外观(Facade)模式
c++·设计模式·外观模式
小小小妮子~16 小时前
框架专题:设计模式
设计模式·框架
先睡16 小时前
MySQL的架构设计和设计模式
数据库·mysql·设计模式
weixin_4231961716 小时前
VSCode+WSL作为IDE开发和管理深度学习项目
ide·vscode·编辑器