C++设计模式:建造者模式(Builder) 房屋建造案例
什么是建造者模式?
建造者模式是一种创建型设计模式,它用于一步步地构建一个复杂对象,同时将对象的构建过程与它的表示分离开。简单来说:
- 它将复杂对象的“建造步骤”分成多部分,让我们可以灵活地控制这些步骤。
- 通过不同的建造者,构建过程可以生成不同的产品。
现实中的例子
想象你在快餐店点套餐:
- 每份套餐包括主食、饮料和甜品。
- 套餐1:汉堡 + 可乐 + 冰淇淋。
- 套餐2:三明治 + 果汁 + 蛋糕。
虽然套餐的结构相同,但具体的内容不同。如果你点套餐时,只需告诉服务员你要套餐1或套餐2,剩下的流程由服务员来完成。这种“组装”套餐的过程就是建造者模式的一个应用场景。
模式的特点
建造者模式特别适合构建那些:
- 由多个部分组成的复杂对象。
- 构建步骤固定但各部分内容可以变化。
- 需要创建不同类型的对象。
模式结构
建造者模式包含以下几个角色:
- 产品(Product):
- 最终构建的复杂对象,由多个部分组成。
- 建造者(Builder):
- 定义对象的构建步骤(如构建地基、搭建结构等)。
- 具体建造者(Concrete Builder):
- 实现具体的构建逻辑,构建特定类型的产品。
- 指挥者(Director):
- 控制建造过程,调用建造者的步骤按顺序完成构建。
- 客户端(Client):
- 选择具体的建造者,通过指挥者完成产品构建,并获取最终产品。
用代码实现一个建造者模式
下面我们以建造房子为例,展示如何使用建造者模式来完成两种房子的构建:木屋 和 玻璃房。
1. 产品类(房子)
房子是一个复杂对象,由地基、结构和屋顶组成。
#include <iostream>
#include <string>
#include <memory> // 用于智能指针管理// 产品类:房子
class House {
public:void setFoundation(const std::string& foundation) {foundation_ = foundation; // 设置地基}void setStructure(const std::string& structure) {structure_ = structure; // 设置结构}void setRoof(const std::string& roof) {roof_ = roof; // 设置屋顶}// 显示房子的组成部分void show() const {std::cout << "房子地基:" << foundation_ << ",结构:" << structure_ << ",屋顶:" << roof_ << "\n";}private:std::string foundation_; // 地基std::string structure_; // 结构std::string roof_; // 屋顶
};
2. 抽象建造者(Builder)
定义房子的建造步骤,包括构建地基、搭建结构和安装屋顶。
// 抽象建造者类
class HouseBuilder {
public:virtual ~HouseBuilder() = default;virtual void buildFoundation() = 0; // 建造地基virtual void buildStructure() = 0; // 建造结构virtual void buildRoof() = 0; // 建造屋顶virtual std::shared_ptr<House> getHouse() = 0; // 返回建造完成的房子
};
3. 具体建造者(木屋和玻璃房)
分别实现木屋和玻璃房的建造逻辑。
// 木屋建造者
class WoodenHouseBuilder : public HouseBuilder {
public:WoodenHouseBuilder() {house_ = std::make_shared<House>();}void buildFoundation() override {house_->setFoundation("木制地基");}void buildStructure() override {house_->setStructure("木制结构");}void buildRoof() override {house_->setRoof("木制屋顶");}std::shared_ptr<House> getHouse() override {return house_;}private:std::shared_ptr<House> house_;
};// 玻璃房建造者
class GlassHouseBuilder : public HouseBuilder {
public:GlassHouseBuilder() {house_ = std::make_shared<House>();}void buildFoundation() override {house_->setFoundation("玻璃地基");}void buildStructure() override {house_->setStructure("玻璃结构");}void buildRoof() override {house_->setRoof("玻璃屋顶");}std::shared_ptr<House> getHouse() override {return house_;}private:std::shared_ptr<House> house_;
};
4. 指挥者类(Director)
指挥者定义房子的建造流程,并调用建造者完成房子。
// 指挥者类
class Director {
public:void setBuilder(std::shared_ptr<HouseBuilder> builder) {builder_ = builder;}// 建造房子的完整流程void constructHouse() {if (builder_) {builder_->buildFoundation();builder_->buildStructure();builder_->buildRoof();}}private:std::shared_ptr<HouseBuilder> builder_;
};
5. 客户端代码
客户端选择建造的房子类型,通过指挥者控制建造流程,最后获取建好的房子。
int main() {Director director; // 创建指挥者// 构建木屋auto woodenBuilder = std::make_shared<WoodenHouseBuilder>();director.setBuilder(woodenBuilder);director.constructHouse();woodenBuilder->getHouse()->show(); // 输出木屋信息// 构建玻璃房auto glassBuilder = std::make_shared<GlassHouseBuilder>();director.setBuilder(glassBuilder);director.constructHouse();glassBuilder->getHouse()->show(); // 输出玻璃房信息return 0;
}
运行结果
程序运行后,输出如下:
房子地基:木制地基,结构:木制结构,屋顶:木制屋顶
房子地基:玻璃地基,结构:玻璃结构,屋顶:玻璃屋顶
模式的优缺点
优点:
- 分离复杂对象的创建过程与表示:
- 客户端只需关心建造步骤,不必关心具体实现。
- 支持多种产品表示:
- 可以通过不同的建造者创建不同的产品。
- 易于扩展:
- 新增产品类型时,只需实现新的建造者类。
缺点:
- 增加复杂性:
- 为每个产品类型都需要定义建造者,代码量可能较多。
- 不适合简单对象的构建:
- 如果对象的结构简单,直接用工厂模式更高效。
适用场景
- 需要创建复杂对象:对象由多个部分组成,并且构建步骤固定。
- 希望支持不同的表示:同样的构建过程可以生成不同的产品(如木屋和玻璃房)。
总结
建造者模式通过将产品的建造过程分解为多个步骤,并定义好构建的流程,使得我们可以灵活地创建不同类型的复杂对象。
它在需要“分步骤创建复杂对象”且“支持多种表示”的场景中非常适用。
本文用构建房子的例子,详细展示了建造者模式的实现过程,希望你能理解并应用这一设计模式!
建造者模式完整程序及详细解释
以下是实现建造者模式的完整程序代码。我们以建造两种房子(木屋和玻璃房)为例,展示如何通过建造者模式分步骤创建复杂对象。
完整代码
#include <iostream>
#include <string>
#include <memory> // 用于智能指针管理// 产品类:房子
class House {
public:void setFoundation(const std::string& foundation) {foundation_ = foundation; // 设置地基}void setStructure(const std::string& structure) {structure_ = structure; // 设置结构}void setRoof(const std::string& roof) {roof_ = roof; // 设置屋顶}// 显示房子的组成部分void show() const {std::cout << "房子地基:" << foundation_ << ",结构:" << structure_ << ",屋顶:" << roof_ << "\n";}private:std::string foundation_; // 地基std::string structure_; // 结构std::string roof_; // 屋顶
};// 抽象建造者类:定义房子建造的步骤
class HouseBuilder {
public:virtual ~HouseBuilder() = default;virtual void buildFoundation() = 0; // 建造地基virtual void buildStructure() = 0; // 建造结构virtual void buildRoof() = 0; // 建造屋顶virtual std::shared_ptr<House> getHouse() = 0; // 返回建造完成的房子
};// 木屋建造者:具体建造者
class WoodenHouseBuilder : public HouseBuilder {
public:WoodenHouseBuilder() {house_ = std::make_shared<House>();}void buildFoundation() override {house_->setFoundation("木制地基");}void buildStructure() override {house_->setStructure("木制结构");}void buildRoof() override {house_->setRoof("木制屋顶");}std::shared_ptr<House> getHouse() override {return house_;}private:std::shared_ptr<House> house_;
};// 玻璃房建造者:具体建造者
class GlassHouseBuilder : public HouseBuilder {
public:GlassHouseBuilder() {house_ = std::make_shared<House>();}void buildFoundation() override {house_->setFoundation("玻璃地基");}void buildStructure() override {house_->setStructure("玻璃结构");}void buildRoof() override {house_->setRoof("玻璃屋顶");}std::shared_ptr<House> getHouse() override {return house_;}private:std::shared_ptr<House> house_;
};// 指挥者类:负责控制建造流程
class Director {
public:void setBuilder(std::shared_ptr<HouseBuilder> builder) {builder_ = builder;}// 按照固定的步骤建造房子void constructHouse() {if (builder_) {builder_->buildFoundation(); // 建造地基builder_->buildStructure(); // 建造结构builder_->buildRoof(); // 建造屋顶}}private:std::shared_ptr<HouseBuilder> builder_;
};// 主函数:客户端代码
int main() {Director director; // 创建指挥者// 使用木屋建造者建造房子auto woodenBuilder = std::make_shared<WoodenHouseBuilder>();director.setBuilder(woodenBuilder);director.constructHouse(); // 按步骤建造木屋woodenBuilder->getHouse()->show(); // 显示木屋信息// 使用玻璃房建造者建造房子auto glassBuilder = std::make_shared<GlassHouseBuilder>();director.setBuilder(glassBuilder);director.constructHouse(); // 按步骤建造玻璃房glassBuilder->getHouse()->show(); // 显示玻璃房信息return 0;
}
运行结果
程序运行后,将输出如下内容:
房子地基:木制地基,结构:木制结构,屋顶:木制屋顶
房子地基:玻璃地基,结构:玻璃结构,屋顶:玻璃屋顶
本文由mdnice多平台发布
相关文章:
C++设计模式:建造者模式(Builder) 房屋建造案例
什么是建造者模式? 建造者模式是一种创建型设计模式,它用于一步步地构建一个复杂对象,同时将对象的构建过程与它的表示分离开。简单来说: 它将复杂对象的“建造步骤”分成多部分,让我们可以灵活地控制这些步骤。通过…...
Python 快速入门(上篇)❖ Python基础知识
Python 基础知识 Python安装**运行第一个程序:基本数据类型算术运算符变量赋值操作符转义符获取用户输入综合案例:简单计算器实现Python安装** Linux安装: yum install python36 -y或者编译安装指定版本:https://www.python.org/downloads/source/ wget https://www.pyt…...
string接口的模拟实现
文章目录 一. string底层逻辑演示声明和定义分开 二. size()三. operator[]四. 迭代器四. const迭代器五. 预留空间(reserve)六. 尾插一个字符push_back七. 尾插一个字符串append八. operator九. operator 一. string底层逻辑 (1)为了和库里面…...
sed使用扩展正则表达式时, -i 要写在 -r 或 -E 的后面
sed使用扩展正则表达式时, -i 要写在 -r 或 -E 的后面 前言 -r 等效 -E , 启用扩展正则表达式 -E是新叫法,更统一,能增强可移植性 , 但老系统,比如 CentOS-7 的 sed 只能用 -r ### Ubuntu24.04-E, -r, --regexp-extendeduse extended regular expressions in the script(fo…...
Verilog HDL可综合与不可综合语句
目录 什么是逻辑综合 可综合语句 不可综合语句 逻辑综合建模建议 综合流程 什么是逻辑综合 所谓逻辑综合就是在标准单元库和特定的设计约束的基础上,把设计的高层次描述转换成优化的门级网表的过程。 标准单元库(工艺库)可以包含简单的…...
tomcat 后台部署 war 包 getshell
1. tomcat 后台部署 war 包 getshell 首先进入该漏洞的文件目录 使用docker启动靶场环境 查看端口的开放情况 访问靶场:192.168.187.135:8080 访问靶机地址 http://192.168.187.135:8080/manager/html Tomcat 默认页面登录管理就在 manager/html 下,…...
网络云计算】2024第47周-每日【2024/11/21】周考-实操题-RAID6实操解析1
文章目录 1、RAID6配置指南(大致步骤)2、注意事项3、截图和视频 网络云计算】2024第47周-每日【2024/11/21】周考-实操题-RAID6实操 RAID6是一种在存储系统中实现数据冗余和容错的技术,其最多可以容忍两块磁盘同时损坏而不造成数据丢失。RAID…...
前端面试题大汇总:React 篇
基础知识 1. 什么是 React?它的主要特点是什么? React 是一个用于构建用户界面的 JavaScript 库,由 Facebook 开发并维护。它主要用于构建单页应用程序(SPA)和复杂的用户界面。React 的主要特点包括: 组件…...
【prism】遇到一个坑,分享!
背景 我通用prism的方式写了一个弹窗,弹窗绑定一个 Loaded 事件,但是Loaded事件一直不触发!!! 具体过程 我的loaded事件也是通过命令的方式绑定的: <i:Interaction.Triggers><i:EventTrigger EventName="Loaded...
Python+Selenium+Pytest+Allure+ Jenkins webUI自动化框架
Python+Selenium+Pytest+Allure+ Jenkins webUI自动化框架 WebUI接口框架使用的工具...
智象未来(HiDream.ai)技术赋能,开启AR眼镜消费时代
Rokid Jungle 2024合作伙伴暨新品发布会于近日隆重举行,标志着AR眼镜跑步进入消费时代,更预示着ARAI技术融合的新篇章。智象未来(HiDream.ai),作为多模态生成式人工智能技术的领跑者,与Rokid的深度合作&…...
element dialog 2层弹窗数据同步问题
注意:本帖为公开技术贴,不得用做任何商业用途 element dialog 2层弹窗数据同步问题 如果嵌套dialog,也就是多层dialog嵌套 2个input,key用同样的值 会导致内外2层dialog,用相同key值的input会数据同步 原因如下&a…...
向量数据库FAISS之五:原理(LSH、PQ、HNSW、IVF)
1.Locality Sensitive Hashing (LSH) 使用 Shingling MinHashing 进行查找 左侧是字典,右侧是 LSH。目的是把足够相似的索引放在同一个桶内。 LSH 有很多的版本,很灵活,这里先介绍第一个版本,也是原始版本 Shingling one-hot …...
要素市场与收入分配
生产要素与家庭收入 生产要素:企业用于生产产品或劳务的最初投入,主要分为三类: 劳动:工人的时间和技能 土地:代指自然资源 资本:指的是货币形式的资本,可以供企业用来购置厂房、设备等资本品…...
Web3的核心技术:区块链如何确保信息安全与共享
在互联网不断迭代的进程中,Web3被视为下一代互联网的核心发展方向,其目标是构建更加开放、安全、去中心化的数字生态。在这一过程中,区块链作为核心技术,为信息安全与共享提供了全新解决方案。本文将深入探讨区块链如何在Web3中实…...
2025蓝桥杯(单片机)备赛--扩展外设之UART1的原理与应用(十二)
一、串口1的实现原理 a.查看STC15F2K60S2数据手册: 串口一在590页,此款单片机有两个串口。 串口1相关寄存器: SCON:串行控制寄存器(可位寻址) SCON寄存器说明: 需要PCON寄存器的SMOD0/PCON.6为0,使SM0和SM1一起指定工作模式,这里选择工作模式1,REN位置1,允许接受, …...
Js中的常见全局函数
文章目录 1、encodeURI、decodeURI2、encodeURIComponent、decodeURIComponent3、parseInt4、parseFloat5、String6、Number7、Boolean8、isNaN、Number.isNaN()9、JSON10、toString Js内置了一些函数和变量,全局都可以获取使用(本文归纳非构造函数作用的…...
MySQL连接查询之自连接
自连接 相当于等值连接,只不过是自己连接自己,不像等值连接是两个不同的表之间的 案例 查询员工名和他的上司的名字 select e.last_name,m.last_name from employees e, employees m #把同一张表当成两张不同表 where e.manager_id m.employee_id;...
Python 爬虫 (1)基础 | 基础操作
一、基础操作 1、快速构建一个爬虫 ConvertCurl: https://curlconverter.com/选择URL,点击右键,选择 Copy >> Copy as cURL(bash) 安装JS环境:https://www.jb51.net/python/307069k7q.htm...
JAVA八股与代码实践----如何为springboot设置Servlet容器为jetty,jetty的优点是什么?
1、实践 排除原来的springboot-web依赖(默认是tomcat),加入jetty的依赖 <dependencies><!-- Spring Boot Web --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-s…...
LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明
LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造,完美适配AGV和无人叉车。同时,集成以太网与语音合成技术,为各类高级系统(如MES、调度系统、库位管理、立库等)提供高效便捷的语音交互体验。 L…...
vue3+vite项目中使用.env文件环境变量方法
vue3vite项目中使用.env文件环境变量方法 .env文件作用命名规则常用的配置项示例使用方法注意事项在vite.config.js文件中读取环境变量方法 .env文件作用 .env 文件用于定义环境变量,这些变量可以在项目中通过 import.meta.env 进行访问。Vite 会自动加载这些环境变…...
ArcGIS Pro制作水平横向图例+多级标注
今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作:ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等(ArcGIS出图图例8大技巧),那这次我们看看ArcGIS Pro如何更加快捷的操作。…...
OPenCV CUDA模块图像处理-----对图像执行 均值漂移滤波(Mean Shift Filtering)函数meanShiftFiltering()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 在 GPU 上对图像执行 均值漂移滤波(Mean Shift Filtering),用于图像分割或平滑处理。 该函数将输入图像中的…...
【Linux】Linux安装并配置RabbitMQ
目录 1. 安装 Erlang 2. 安装 RabbitMQ 2.1.添加 RabbitMQ 仓库 2.2.安装 RabbitMQ 3.配置 3.1.启动和管理服务 4. 访问管理界面 5.安装问题 6.修改密码 7.修改端口 7.1.找到文件 7.2.修改文件 1. 安装 Erlang 由于 RabbitMQ 是用 Erlang 编写的,需要先安…...
在 Visual Studio Code 中使用驭码 CodeRider 提升开发效率:以冒泡排序为例
目录 前言1 插件安装与配置1.1 安装驭码 CodeRider1.2 初始配置建议 2 示例代码:冒泡排序3 驭码 CodeRider 功能详解3.1 功能概览3.2 代码解释功能3.3 自动注释生成3.4 逻辑修改功能3.5 单元测试自动生成3.6 代码优化建议 4 驭码的实际应用建议5 常见问题与解决建议…...
热门Chrome扩展程序存在明文传输风险,用户隐私安全受威胁
赛门铁克威胁猎手团队最新报告披露,数款拥有数百万活跃用户的Chrome扩展程序正在通过未加密的HTTP连接静默泄露用户敏感数据,严重威胁用户隐私安全。 知名扩展程序存在明文传输风险 尽管宣称提供安全浏览、数据分析或便捷界面等功能,但SEMR…...
土建施工员考试:建筑施工技术重点知识有哪些?
《管理实务》是土建施工员考试中侧重实操应用与管理能力的科目,核心考查施工组织、质量安全、进度成本等现场管理要点。以下是结合考试大纲与高频考点整理的重点内容,附学习方向和应试技巧: 一、施工组织与进度管理 核心目标: 规…...
CppCon 2015 学习:Time Programming Fundamentals
Civil Time 公历时间 特点: 共 6 个字段: Year(年)Month(月)Day(日)Hour(小时)Minute(分钟)Second(秒) 表示…...
密码学基础——SM4算法
博客主页:christine-rr-CSDN博客 专栏主页:密码学 📌 【今日更新】📌 对称密码算法——SM4 目录 一、国密SM系列算法概述 二、SM4算法 2.1算法背景 2.2算法特点 2.3 基本部件 2.3.1 S盒 2.3.2 非线性变换 编辑…...
