实战设计模式之建造者模式
概述
在实际项目中,我们有时会遇到需要创建复杂对象的情况。这些对象可能包含多个组件或属性,而且每个组件都有自己的配置选项。如果直接使用构造函数或前面介绍的工厂方法来创建这样的对象,可能会导致以下两个严重问题。
1、参数过多。当一个类有很多可选参数时,构造函数可能会变得非常庞大,难以维护。此外,调用方必须记住哪些参数是必需的,哪些是可选的,这在一定程度上增加了出错的风险。
2、代码冗余。如果每次创建对象都需要传递大量的参数,那么代码将会变得冗长、复杂且难以理解。即使有些参数有默认值,调用者仍然需要为所有参数提供值。
相比之下,建造者模式提供了更好的解决方案。它通过引入一个专门的建造者类来逐步构建对象,直到所有的必要部分都被设置完毕。这样不仅可以简化对象的创建过程,还可以提高代码的可读性和可维护性。
餐厅点餐是运用建造者模式的一个典型例子:菜单上有多种套餐组合,但每个套餐的具体内容可能不同;服务员会根据我们的选择,逐步记录下我们想要的食物和饮料,最后再把这些信息传递给厨房。在这个过程中,服务员扮演了“导演”的角色,而每道菜的厨师则是具体的“建造者”,他们负责按照订单的要求制作食物。
基本原理
建造者模式的核心思想是:将一个复杂对象的构建过程与其表示分离,从而使得同样的构建过程可以创建不同的表示。它提供了一种分步骤构造复杂对象的方法,避免了大型构造函数带来的问题,并且提高了代码的可读性和可维护性。建造者模式包括如下四个核心组件。
1、产品类。代表要创建的复杂对象,它可以是一个具体的产品实例,也可以是一系列相关属性的集合。产品类通常包含多个组成部分,每个部分都有自己的配置选项。
2、抽象建造者。定义了创建产品各个部分的接口,它是所有具体建造者的基类或接口,声明了一系列用于组装产品的抽象方法。这些方法定义了如何构建产品的不同部分,但不涉及具体的实现逻辑。
3、具体建造者。实现了抽象建造者接口,提供了构建产品的具体实现。每个具体建造者都对应一种特定的产品配置或类型,负责按照预定规则组装出完整的产品。通常情况下,具体建造者还会提供一个类似GetResult的方法来获取最终构建的产品对象。
4、导演类。导演类负责协调各个具体建造者的执行顺序,它不直接参与产品的构建,而是调用建造者的方法来组织构建过程。它可以根据不同的需求选择合适的建造者,并控制构建的流程。
基于上面的核心组件,建造者模式的实现主要有以下五个步骤。
1、定义产品类。创建一个代表最终产品的类,它包含了所有与产品相关的属性。
2、定义抽象建造者。定义一个接口或抽象类,声明一组用于构建产品的方法。这些方法定义了如何设置产品的各个部分,但不涉及具体的实现细节。
3、实现具体建造者。根据需要创建多个具体建造者类,每个类都实现了抽象建造者接口。每个具体建造者负责按照特定的规则,组装出完整的产品,并提供GetResult方法来返回最终构建的产品对象。
4、创建导演类。定义一个导演类,它负责调用具体建造者的方法来组织构建过程。导演类不直接参与产品的构建,而是充当协调者的角色,确保构建过程按照预期进行。
5、编写应用层代码。在应用层代码中,首先选择合适的具体建造者并将其传递给导演类。导演类会根据预设的构建流程调用建造者的方法,逐步构建出最终的产品对象。最后,应用层可以从建造者那里获取到完全配置好的产品实例。
实战解析
在下面的实战代码中,我们使用建造者模式模拟了餐厅点餐的应用场景。
首先,我们定义了产品类COrder。COrder类代表最终的订单对象,它包含了主菜、饮料、甜点以及额外添加的项目。
然后,我们定义了抽象建造者COrderBuilder。COrderBuilder是一个接口或抽象类,声明了一系列用于设置订单各个部分的方法。每个具体建造者都将实现这些方法,以按照特定规则组装出完整的订单。
接下来,我们实现了具体建造者CStandardMealBuilder和CVegetarianMealBuilder,分别表示创建标准套餐和素食套餐。每个具体建造者都实现了COrderBuilder接口,并提供了构建特定类型订单的具体逻辑。
我们还创建了导演类CWaiter,它负责协调各个建造者的执行顺序。它不直接参与订单的构建,而是调用建造者的方法来组织构建过程,这样可以确保构建过程的一致性和灵活性。
最后,在main函数中,我们首先选择了CStandardMealBuilder来构建一个标准套餐,并通过CWaiter类来指导构建过程。紧接着,我们选择了CVegetarianMealBuilder来构建一个素食套餐。最后,我们打印了两个订单的具体内容,并进行了清理工作。
#include <iostream>
#include <string>
#include <vector>using namespace std;// 定义产品类:订单对象
class COrder
{
public:void SetMainDish(const string& strDish){m_strMainDish = strDish;}void SetDrink(const string& strDrink){m_strDrink = strDrink; }void SetDessert(const string& strDessert){m_strDessert = strDessert;}void AddExtra(const string& strExtra){m_vctExtras.push_back(strExtra);}void Print() const{cout << "Your order contains: " << endl;if (!m_strMainDish.empty()){cout << "Main dish: " << m_strMainDish << endl;}if (!m_strDrink.empty()){cout << "Drink: " << m_strDrink << endl;}if (!m_strDessert.empty()){cout << "Dessert: " << m_strDessert << endl;}if (!m_vctExtras.empty()){cout << "Extras: " << endl;for (const auto& extra : m_vctExtras){cout << " " << extra << endl;}}}private:string m_strMainDish; // 主菜string m_strDrink; // 饮料string m_strDessert; // 甜点vector<string> m_vctExtras; // 额外项
};// 定义抽象建造者
class COrderBuilder
{
public:COrderBuilder(){m_pOrder = new COrder();}virtual ~COrderBuilder() {}virtual void BuildMainDish() = 0;virtual void BuildDrink() = 0;virtual void BuildDessert() = 0;virtual void AddExtras() = 0;virtual COrder* GetResult(){return m_pOrder; }protected:COrder* m_pOrder;
};// 实现具体建造者:标准套餐建造者
class CStandardMealBuilder : public COrderBuilder
{
public:void BuildMainDish() override{m_pOrder->SetMainDish("Roast Chicken");}void BuildDrink() override{m_pOrder->SetDrink("Cola");}void BuildDessert() override{m_pOrder->SetDessert("Ice Cream");}void AddExtras() override{m_pOrder->AddExtra("Potato Fries");}
};// 实现具体建造者:素食套餐建造者
class CVegetarianMealBuilder : public COrderBuilder
{
public:void BuildMainDish() override{m_pOrder->SetMainDish("Vegetable Curry");}void BuildDrink() override{m_pOrder->SetDrink("Lemonade");}void BuildDessert() override{m_pOrder->SetDessert("Fruit Salad");}void AddExtras() override{m_pOrder->AddExtra("Breadsticks");}
};// 创建导演类:服务员
class CWaiter
{
public:void Construct(COrderBuilder* pBuilder){pBuilder->BuildMainDish();pBuilder->BuildDrink();pBuilder->BuildDessert();pBuilder->AddExtras();}
};int main()
{CWaiter waiter;// 选择标准套餐COrderBuilder* pBuilder = new CStandardMealBuilder();waiter.Construct(pBuilder);COrder* pStandardOrder = pBuilder->GetResult();cout << "***** Standard Meal *****" << endl;pStandardOrder->Print();cout << endl;delete pBuilder;// 选择素食套餐pBuilder = new CVegetarianMealBuilder();waiter.Construct(pBuilder);COrder* pVegetarianOrder = pBuilder->GetResult();cout << "***** Vegetarian Meal *****" << endl;pVegetarianOrder->Print();delete pBuilder;// 清理delete pStandardOrder;delete pVegetarianOrder;return 0;
}
总结
对于那些需要很多参数的对象创建,尤其是这些参数大多数是可选的时候,使用建造者模式可以让代码更加易读、易于维护。相比长串的构造函数参数列表,建造者模式提供了更自然的接口来逐步设定属性。
但建造者模式也引入了额外的类(即具体的建造者),因此会稍微增加系统的复杂度和整体大小。对于简单的对象创建,这种模式可能显得过于繁琐。如果要添加新的产品,就需要修改现有的建造者接口或创建新的建造者子类,这可能会影响现有代码的稳定性。另外,如果建造者负责多个不同类型的对象创建,可能会导致建造者类变得臃肿,承担过多责任,从而违背了单一职责原则。
相关文章:

实战设计模式之建造者模式
概述 在实际项目中,我们有时会遇到需要创建复杂对象的情况。这些对象可能包含多个组件或属性,而且每个组件都有自己的配置选项。如果直接使用构造函数或前面介绍的工厂方法来创建这样的对象,可能会导致以下两个严重问题。 1、参数过多。当一个…...

活动预告 | Microsoft Azure 在线技术公开课:使用 Azure OpenAI 服务构建生成式应用
课程介绍 通过 Microsoft Learn 免费参加 Microsoft Azure 在线技术公开课,掌握创造新机遇所需的技能,加快对 Microsoft Cloud 技术的了解。参加我们举办的“使用 Azure OpenAI 服务构建生成式应用”活动,了解如何使用包括 GPT 在内的强大的…...

ubuntu安装firefox
firefox下载地址:https://ftp.mozilla.org/pub/firefox/releases/ 卸载 sudo apt-get update dpkg --get-selections |grep firefox apt-get purge firefox 解压 tar -xjf firefox*.tar.bz2复制文件 sudo mv firefox/ /opt/firefox30sudo mv /usr/bin/firefox /…...

计算机网络原理(谢希仁第八版)第4章课后习题答案
第四章 网络层 详细计算机网络(谢希仁-第八版)第四章习题全解_计算机网络第八版谢希仁课后答案-CSDN博客 1.网络层向上提供的服务有哪两种?是比较其优缺点。网络层向运输层提供 “面向连接”虚电路(Virtual Circuit)服…...

RabbitMQ-基本使用
RabbitMQ: One broker to queue them all | RabbitMQ 官方 安装到Docker中 docker run \-e RABBITMQ_DEFAULT_USERrabbit \-e RABBITMQ_DEFAULT_PASSrabbit \-v mq-plugins:/plugins \--name mq \--hostname mq \-p 15672:15672 \-p 5672:5672 \--network mynet\-d \rabbitmq:3…...

从零开始学架构——互联网架构的演进
1 技术演进 1.1 技术演进的动力 对于新技术,我们应该站在行业的角度上思考,哪些技术我们要采取,哪些技术我们不能用,投入成本过大会不会导致满盘皆输?市场、技术、管理三者组成的业务发展铁三角,任何一个…...

python +tkinter绘制彩虹和云朵
python tkinter绘制彩虹和云朵 彩虹,简称虹,是气象中的一种光学现象,当太阳光照射到半空中的水滴,光线被折射及反射,在天空上形成拱形的七彩光谱,由外圈至内圈呈红、橙、黄、绿、蓝、靛、紫七种颜色。事实…...

重新整理机器学习和神经网络框架
本篇重新梳理了人工智能(AI)、机器学习(ML)、神经网络(NN)和深度学习(DL)之间存在一定的包含关系,以下是它们的关系及各自内容,以及人工智能领域中深度学习分支对比整理。…...

TypyScript从入门到精通
TypyScript从入门到精通 TypyScript 是什么?增加了什么环境搭建二、为何需要 TypeScript三、编译 TypeScript四、类型声明五、类型推断基本类型六、类型总览JavaScript 中的数据类型TypeScript 中的数据类型1. 上述所有 JavaScript 类型2. 六个新类型:3.…...

【MATLAB】绘制投资组合的有效前沿
文章目录 一、数据准备二、有效前沿三、代码3.1 数据批量读取、预处理3.2 绘制可行集3.3 绘制有效前沿3.4 其它-最大夏普率 一、数据准备 准备多个股票的的历史数据,目的就是找到最优的投资组合。 下载几个标普500里面的公式的股票数据吧,下载方法也可…...

matlab时频分析库
time frequency gallery...

GBase 8s 数据库备份还原
每一天都是一个新的篇章,等待着你去书写属于自己的故事!!! 一:备份 1.1.下载脚本文件,并上传到数据库服务器上相应目录。 解压后目录为: 说明: dbcomm.sh:导出注释脚本…...

C++模板相关概念汇总
文章目录 一、模板的概念与作用二、函数模板模板的非类型参数调用顺序 三、类模板四、模板的编译模型 一、模板的概念与作用 C模板是一种强大的代码复用机制,它允许程序员编写通用的代码,能够处理不同类型的数据,而无需为每种类型都重复编写…...

MYSQL------sql基础
SQL基础与简介 定义:SQL即结构化查询语言(Structured Query Language),是一种特殊目的的编程语言,用于存取数据以及查询、更新和管理关系数据库系统。作用:可以用于数据库的创建、数据的插入、查询、更新和…...

React Router 用法概览
React Router 用法 React 使得开发者能够轻松地创建交互式的单页应用(SPA),单页应用的一个常见挑战是如何处理页面导航和路由吗,React Router 就是解决这个问题的工具 路由(Router)是 React Router 的核心…...

网络安全之高防IP的实时监控精准防护
高防IP是一种网络安全设备,用于保护网络服务不受到各类攻击的影响,确保业务的持续稳定运行。它通过监控、识别和封锁恶意攻击流量,提供高级别的防护,降低业务被攻击的风险,并提升网络的稳定性和可靠性。 一、实时监控的…...

2024年中国新能源汽车用车发展怎么样 PaperGPT(二)
用车趋势深入分析 接上文,2024年中国新能源汽车用车发展怎么样 PaperGPT(一)-CSDN博客本文将继续深入探讨新能源汽车的用车强度、充电行为以及充电设施的现状。 用车强度 月均行驶里程:2024年纯电车辆月均行驶超过1500公里&…...

LINUXC 时间相关操作
文章目录 时间戳获取本地时间struct tm 结构体高精度的时间struct timeval 结构体相关函数time()localtime()gmtime()gettimeofday()strftime()mktime() 示例代码 时间戳 时间戳是指计算机中存储的数字型时间。它是以一个特定的时间点作为起点(通常是1970年1月1日0…...

网络游戏之害
网络游戏之害: 网络游戏于今之世风靡四方,其娱人耳目、畅人心怀之效,固为人知,然所藏之害,若隐伏之暗潮,汹涌而至时,足以覆舟,尤以青年为甚,今且缕析其害,以…...

SpringMVC的消息转换器
SpringMVC的消息转换器(Message Converter)是Spring框架中用于处理HTTP请求和响应体与Java对象之间转换的组件。它们使得开发人员可以轻松地将HTTP请求的数据映射到方法参数,并将返回的对象转换为HTTP响应。 工作原理 当一个HTTP请求到达Spr…...

Chrome 浏览器下载安装教程,保姆级教程
大家好,今天我们来聊一聊如何在国内下载和安装最新版本的 Chrome 浏览器。由于众所周知的原因,Google 的网站在国内是被屏蔽的,因此很多朋友在下载 Chrome 浏览器 时会遇到困难。其实,不必担心,今天我将为大家带来一份…...

ElasticSearch系列(一)
一.了解ES、倒排索引、es的一些概念、安装es、kibana 二.DSL;索引库操作 三.Java RestClient:索引库操作 一、了解ES、倒排索引、es的一些概念、安装es、kibana kibana、logstash、beats Elasticserach 存储,计算 ,搜索数据 –…...

C++技巧:map和vector
一,map是有序的,unordered_map是无序的 在C中,std::map 和 std::unordered_map 是两种不同的容器,它们都用于存储键值对,但是它们的底层实现和性能特性有所不同。以下是它们的详细介绍: std::map std::m…...

中建海龙:科技助力福城南产业片区绿色建筑发展
在快速发展的城市化进程中,绿色建筑以其环保、节能、可持续的特点日益受到重视。作为建筑工业化领域的领军企业,中建海龙科技有限公司(简称“中建海龙”)凭借其卓越的科技实力和创新举措,在推动绿色建筑发展方面做出了…...

模块化通讯管理机在物联网系统中的应用
安科瑞刘鸿鹏 摘要 随着能源结构转型和智能化电网的推进,电力物联网逐渐成为智能电网的重要组成部分。本文以安科瑞ANet系列智能通信管理机为例,探讨其在电力物联网中的应用,包括数据采集、规约转换、边缘计算、远程控制等技术实践&#…...

建立一个Macos载入image的实例含界面
前言 为了方便ios程序的开发,有时候需要先用的Macos平台进行一些功能性的程序开发。 作为对比和参考。 1、创建一个MacOS的App 2、主界面控件的增加 添加的控件方法与ios相同,也是再用commandshiftL(CtrlShiftL),就会弹出控件…...

Redis List列表
个人主页:C忠实粉丝 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 C忠实粉丝 原创 Redis List列表 收录于专栏[redis] 本专栏旨在分享学习Redis的一点学习笔记,欢迎大家在评论区交流讨论💌 目录 概述 常用命令 LPUSH …...

继承与多态 - 继承机制、虚函数、纯虚函数
引言 C 是一种支持面向对象编程(OOP)的编程语言,继承和多态是 OOP 的两个核心概念。通过继承,我们可以创建新的类,这些新类可以重用现有类的代码,并且可以根据需要进行扩展或修改。多态则允许我们编写更加…...

【QT】C++线程安全的单例模板
模板代码 #pragma once #include <mutex> #include <atomic>// CRTP基类模板 Curiously Recurring Template Parttern—奇异递归模板模式。 template <typename T> class SingletonCRTP { public:// 禁止拷贝构造和赋值操作SingletonCRTP(const SingletonCR…...

node.js内置模块之---EventEmitter 类
EventEmitter 类什么作用 EventEmitter 类的主要方法 EventEmitter 类什么作用 在 Node.js 中,EventEmitter 是一个非常核心的类,它提供了一种事件驱动的机制。几乎所有的 Node.js 核心模块(如 fs, http, net 等)都采用了事件驱…...