[设计模式] 建造者模式

news2024/9/29 9:36:41

文章目录

  • 什么是建造者模式
  • 建造者模式
    • 建造者模式中的角色
    • UML类图
    • 代码实现
  • 建造者模式与工厂模式的区别

什么是建造者模式

建造者模式(Builder Pattern)是一种创建型的设计模式,它将一个复杂对象的构建与它的表示分离,也就是复杂的构建隐藏起来,并且加以抽象。

它的好处是建造者独立,易扩展。而且便于控制细节风险。缺点是产品必须有共同点,范围有限制。如内部变化复杂,会有很多的建造类。

建造者模式

建造者模式中的角色

建造者模式中的所有角色如下:

  • 产品(Product):产品是被构建的复杂对象,包含多个组成部件,具体建造者创建该产品的内部表示并定义它的装配过程。
  • 抽象建造者(Builder):抽象建造者为创建一个产品对象的各个部件指定抽象接口。
  • 具体建造者(ConcreteBuilder):具体建造者实现了抽象建造者接口,实现各个部件的构造和装配方法,定义并明确它所创建的复杂对象,也可以提供一个方法返回创建好的复杂产品对象。
  • 指挥者(Director):指挥者负责安排复杂对象的建造次序,指挥者与抽象建造者之间存在关联关系,可调用建造者对象的部件与装配方法,完成发展对象的建造。

UML类图

在这里插入图片描述

代码实现

产品类:

class ProductComputer{
private:
	string m_CPU;
	string m_RAM;
	string m_DISK;
public:
	ProductComputer(){
		cout<<"Constructor of Product"<<endl;
	}
	~ProductComputer(){
		cout<<"Destructor of Product"<<endl;
	}
public:
	void setCPU(string CPU){ 
		this->m_CPU = CPU;
	}
	void setRAM(string RAM){
		this->m_RAM = RAM;
	}
	void setDISK(string DISK){
		this->m_DISK = DISK; 
	}
	void showHW(){
		/* definition of split line:
			#define     SPLIT_LINE "============================="
			#define SUB_SPLIT_LINE "-----------------------------"
		*/
		cout<<SPLIT_LINE<<"\nComputer Hardware List:\n"<<SUB_SPLIT_LINE<<"\nCPU:  "<<m_CPU<<"\nRAM:  "<<m_RAM<<"\nDISK: "<<m_DISK<<endl<<SPLIT_LINE<<endl;
	}
};

其中在输出的时候,为了使输出更加清晰,我们定义了分隔线SPLIT_LINESUB_SPLIT_LINE,这个并不是必要的,你可以选择去掉:

#define     SPLIT_LINE "============================="
#define SUB_SPLIT_LINE "-----------------------------"

创建者接口(抽象创建者):

class IBuilder
{
public:
    ProductComputer *m_product;
public:
	IBuilder(){
		cout<<"Constructor of Interface Builder"<<endl;
		m_product = new ProductComputer();
	}
	~IBuilder(){
		cout<<"Destructor of Interface Builder"<<endl;
		if(m_product != nullptr){
			delete m_product;
			m_product = nullptr;
		}
	}
public:
    virtual void buildCPU () = 0;
    virtual void buildRAM () = 0;
    virtual void buildDISK() = 0;
    virtual ProductComputer* getProductPtr() = 0;
};

具体创建者:

class ConcreteBuilder: public IBuilder{
public:
	ConcreteBuilder(){
		cout<<"Constructor of ConcreteBuilder"<<endl;
	}
	~ConcreteBuilder(){
		cout<<"Destructor of ConcreteBuilder"<<endl;
	}
public:
	void buildCPU(){
		this->m_product->setCPU("Intel Core i7-13700KF");
	}
	void buildRAM(){
		this->m_product->setRAM("Kingston FURY 32GB DDR5");
	}
	void buildDISK(){
		this->m_product->setDISK("WD SSD M.2(NVMe) 500G");
	}
	ProductComputer* getProductPtr(){
		return m_product;
	}
};

指挥者类:

class Director{
private:
	IBuilder* m_builder;
public:
	Director(){
		cout<<"Constructor of Director"<<endl;
	}
	~Director(){
		cout<<"Destructor of Director"<<endl;
	}
public:
	void setBuilder(IBuilder* builder){
		this->m_builder = builder;
	}
	ProductComputer* construct(){
		m_builder->buildCPU();
		m_builder->buildRAM();
		m_builder->buildDISK();
		return m_builder->getProductPtr();
	}
};

主函数:

int main(){
	Director *director = new Director();
	IBuilder* builder = new ConcreteBuilder();
	director->setBuilder(builder);

	ProductComputer* computer = director->construct();
	computer->showHW();

	delete director;
	delete (ConcreteBuilder*)builder;//to free the interface and conreatebuilder
}

Notice
这里一定要注意对于创建者delete时需要使用下面这样:

delete (ConcreteBuilder*)builder;

否则的话只会析构Ibuilder而不会析构ConcreteBuilder

输出如下:

Constructor of Director
Constructor of Interface Builder
Constructor of Product
Constructor of ConcreteBuilder
=============================
Computer Hardware List:
-----------------------------
CPU:  Intel Core i7-13700KF
RAM:  Kingston FURY 32GB DDR5
DISK: WD SSD M.2(NVMe) 500G
=============================
Destructor of Director
Destructor of ConcreteBuilder
Destructor of Interface Builder
Destructor of Product

建造者模式与工厂模式的区别

在看完上面代码后,我们应该对建造者模式有了一定了解,我们可以看出:建造者模式更加关注与零件装配的顺序。然而工厂模式不需要关注这么多,只需创建实例即可。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/339194.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

即时通讯系列-4-如何设计写扩散下的同步协议方案

1. 背景信息 上篇提到了, IM协议层是主要解决会话和消息的同步, 在实现上, 以推模式为主, 拉模式为辅. 本文Agenda: (How)如何同步(How)如何设计同步位点如何设计 Gap过大(SyncGapOverflow) 机制如何设计Ack机制总结 提示: 本系列文章不会单纯的给出结论, 希望能够分享的是&…

SpringCloud-Netflix学习笔记13——Zuul路由网关

什么是Zuul? Zuul包含了对请求的路由和过滤两个最主要的功能。 其中路由功能负责将外部请求转发到具体的微服务实例上&#xff0c;是实现外部访问统一入口的基础&#xff0c;而过滤器功能则负责对请求的处理过程进行干预&#xff0c;是实现请求校验&#xff0c;服务聚合等功能…

最详细教你注册 ChatGPT,不会来找我

超强人工智能 ChatGPT 震撼来袭&#xff0c;它是美国人工智能研究实验室 OpenAI 新推出的一种自然语言处理工具&#xff0c;不想来体验一下嘛&#xff01;最详细教程手把手教你注册&#xff0c;不会来找我&#xff01; 准备工作 一个可以科学上网的工具&#xff0c;提供非 Ch…

文献阅读笔记 # CodeBERT: A Pre-Trained Model for Programming and Natural Languages

《CodeBERT: A Pre-Trained Model for Programming and Natural Languages》EMNLP 2020 (CCF-B)作者主要是来自哈工大、中山大学的 MSRA 实习生和 MSRA、哈工大的研究员。资源&#xff1a;code | pdf相关资源&#xff1a;RoBERTa-base | CodeNN词汇&#xff1a; bimodal: 双模态…

嵌入式设备搭建NFS环境(服务器/客户端、源码下载编译、文件系统适配、内核适配)

1、什么是nfs (1)NFS(Network File System)是网络文件系统&#xff0c;能让使用者访问网络上别处的文件就像在使用自己的计算机一样&#xff1b; (2)NFS是基于UDP/IP协议的应用&#xff0c;其实现主要是采用远程过程调用RPC机制&#xff0c;RPC提供了一组与机器、操作系统以及低…

CAS详解.

CAS这个机制就给实现线程安全版本的代码&#xff0c;提供了一个新的思路&#xff0c;之前通过加锁&#xff0c;把多个指令打包成整体&#xff0c;来实现线程安全。现在就可以考虑直接基与CAS来实现一些修改操作&#xff0c;也能保证线程安全&#xff08;不需要加锁&#xff09;…

OpenAi-chatgpt注册保姆级全网最详细注册教程2023年2月最新-

废话就不多说了&#xff0c;说多了浪费各位师傅的时间&#xff01;直接冲&#xff0c;在开始之前需要科学上网&#xff0c;就没其他要求了 1、访问https://chat.openai.com/auth/login 2、点击sign up,输入账号密码&#xff0c;点击Continue 3、之后会来到登陆页面&#xff0…

Oracle Dataguard(主库为 Oracle rac 集群)配置教程(03)—— 创建 dataguard 数据库之前的准备工作

Oracle Dataguard&#xff08;主库为 Oracle rac 集群&#xff09;配置教程&#xff08;03&#xff09;—— 创建 dataguard 数据库之前的准备工作 / 本专栏详细讲解 Oracle Dataguard&#xff08;Oracle 版本为11g&#xff0c;主库为双节点 Oracle rac 集群&#xff09;的配置…

云计算|OpenStack|错误记录和解决方案(不定时更新)

前言&#xff1a; openstack的部署和使用是难度比较大的&#xff0c;难免会出现各种各样的问题&#xff0c;因此&#xff0c;本文将把一些在部署和使用openstack社区版时出现的错误做一个记录&#xff0c;并就每一个错误分析和解决问题。&#xff08;尽量记录比较经典的错误&a…

微搭低代码从入门到精通10-tab栏组件

在小程序中&#xff0c;如果你的页面是由多个组成的&#xff0c;往往涉及到页面切换的问题。那如何引导用户访问不同的页面呢&#xff1f;微搭中提供了tab栏组件来实现这个功能&#xff0c;本篇我们介绍一下这个组件的使用方法。 首先呢打开我们的应用编辑器&#xff0c;在左侧…

OJ刷题Day2 · 判断根结点是否等于子结点之和 · 删除有序数组中的重复项 · 合并两个有序链表 · 数组中的第K个最大元素(中等题)

一、判断根结点是否等于子结点之和二、删除有序数组中的重复项三、合并两个有序链表四、数组中的第K个最大元素&#xff08;中等题&#xff09;一、判断根结点是否等于子结点之和 给你一个 二叉树 的根结点 root&#xff0c;该二叉树由恰好 3 个结点组成&#xff1a;根结点、左…

SpringBoot + kotlin/java + Mybatis-Plus +Sqlite + Gradle多模块项目

前言 我自己的业务项目&#xff0c;先用kotlinspringboot 搭建&#xff0c; 发现gradle支持kts脚本&#xff0c;于是我就搭建试试。我就选用了最流行的Sqlite内嵌数据库,虽然H2也不错&#xff0c;但是Sqlite才是最流行的。orm框架我还是选择了Mybatis-Plus &#xff0c;为此中…

Spring Boot的创建和使用

目录 一、Spring Boot介绍 1.1 Spring Boot 是什么 1.2 Spring Boot的优点 二、Spring Boot 项目的创建 2.1 使用idea创建 2.1.1 安装Spring Boot Helper插件 2.1.2 创建 Spring Boot 项目 2.1.3 验证项目是否创建成功 2.2 使用网页创建 三、输出 hello world 一、S…

前端如何提升To B产品用户体验

云计算产品发展的早期常以技术为核心吸引客户&#xff0c;功能的实现是这一时期产品优先考虑的因素。经过数十年的发展&#xff0c;云计算行业已经进入了深耕细作的时代&#xff0c;市场的激烈竞争与云产品快速发展的同时&#xff0c;用户对产品的可用性与易用性也有了更高的要…

交换机中的冗余链路管理

一 交换机冗余链路许多交换机或交换机设备组成的网络环境中&#xff0c;通常使用一些备份连接&#xff0c;以提高网络的健全性&#xff0c;稳定性。备份连接也叫备份链路&#xff0c;冗余链路等。为了解决共享式局域网的碰撞问题&#xff0c;采用了交换机构成的交换式局域网&am…

C语言静态库、动态库的封装和注意事项

1、动态库、静态库介绍 参考博客&#xff1a;《静态库和动态库介绍以及Makefile》&#xff1b; 2、代码目录结构和编译脚本 参考博客&#xff1a;《实际工作开发中C语言工程的目录结构分析》&#xff1b; 3、编写库的流程 (1)明确需求:需求是否合理、需求的使用场景、需求可能遇…

chatgpt:人工智能的一次突破,如何正确的创建用户及使用

Chatgpt的正确创建及使用 chatgpt最近在国内也开始有声音了&#xff0c;其实早在去年12月初&#xff0c;该网站就已经可以在国外进行使用&#xff0c;而且很快渗透到了国外各行各业各个年龄段 &#xff0c;最火的当属国外很多学生用它来生成论文&#xff0c;关键是语句通顺&am…

如何开启多个独立Chrome浏览器

一、简介 作为测试或者开发人员&#xff0c;有些情况下会用到 Chrome 浏览器&#xff0c;但有时是同一个 Chrome 浏览器无法为我们提供隔离开的不同环境。这样 我们就需要清理 cache 、切换账号等&#xff0c;降低了我们的工作效率。今天的主题是如何开启多个独立的 Chrome 浏…

LayUI模板引擎渲染数据

前端模板引擎介绍 接上节Spring boot项目开发实战——&#xff08;LayUI实现前后端数据交换与定义方法渲染数据&#xff09; 模板引擎能简化开发&#xff0c;极大提高效率&#xff0c;小编之前使用过JSP和Thymeleaf&#xff0c;以及python的jinja2&#xff0c;这些是后端的模…

spring(二)-----------如何注入bean

我们从第三方框架mybatis为引&#xff0c;看看如何往spring中注入一个bean 1、纯mybatis开发生成一个mapper对象 如果不使用spring的情况下&#xff0c;mybatis想生成一个mapper对象大概需要做下面的操作&#xff1a; 假设我们有了一个TMapper接口&#xff0c;此时获取该map…