「软件设计模式」建造者模式(Builder)

深入解析建造者模式:用C++打造灵活对象构建流水线

引言:当对象构建遇上排列组合

在开发复杂业务系统时,你是否经常面对这样的类:它有20个成员变量,其中5个是必填项,15个是可选项。当用户需要创建豪华套餐A(含加冰可乐)基础套餐B(不要洋葱)时,传统的构造函数早已不堪重负。这正是建造者模式(Builder Pattern)大展拳脚的舞台!

一、模式精髓解析

建造者模式通过分步构建的方式解耦复杂对象的创建过程,让同一构建流程能产出不同表现形式的产品。就像麦当劳的点餐系统:选择汉堡基底→添加配料→选择饮料→完成套餐组合。

二、C++模式实现结构

核心组件拆解

  1. Product(产品):需要构建的复杂对象
  2. Builder(抽象建造者):定义构建步骤的接口
  3. ConcreteBuilder(具体建造者):实现具体构建逻辑
  4. Director(指挥者):控制构建流程(可选)

三、实战:定制汉堡套餐系统

我们以快餐店汉堡套餐系统为例,演示如何实现不同配置的套餐组合。

1. 产品类定义

#include <iostream>
#include <memory>
#include <string>
#include <vector>class HamburgerMeal {
public:void showMeal() const {std::cout << "=== 您的套餐配置 ===" << std::endl;std::cout << "主餐: " << mainItem << std::endl;std::cout << "饮料: " << drink << std::endl;std::cout << "附加项: ";for (const auto& item : sides) {std::cout << item << " ";}std::cout << "\n甜点: " << (dessert.empty() ? "无" : dessert) << std::endl;}private:friend class MealBuilder; // 允许建造者访问私有成员std::string mainItem;std::string drink;std::vector<std::string> sides;std::string dessert;
};

2. 建造者实现

class MealBuilder {
public:MealBuilder() = default;MealBuilder& setMain(const std::string& main) {meal.mainItem = main;return *this;}MealBuilder& setDrink(const std::string& drink) {meal.drink = drink;return *this;}MealBuilder& addSide(const std::string& side) {meal.sides.push_back(side);return *this;}MealBuilder& setDessert(const std::string& dessert) {meal.dessert = dessert;return *this;}HamburgerMeal build() {// 验证必要参数if (meal.mainItem.empty()) {throw std::invalid_argument("必须指定主餐");}return meal;}private:HamburgerMeal meal;
};

3. 客户端调用

#include "../../include/header.h"
#include "buger.h"
#include "meal_builder.h"
int main() {try {// 豪华套餐HamburgerMeal premiumMeal = MealBuilder().setMain("安格斯牛肉汉堡").setDrink("大杯可乐").addSide("薯条").addSide("鸡块").setDessert("苹果派").build();// 简约套餐HamburgerMeal simpleMeal = MealBuilder().setMain("经典鸡腿堡").setDrink("小杯雪碧").build();premiumMeal.showMeal();std::cout << "\n";simpleMeal.showMeal();} catch (const std::exception& e) {std::cerr << "套餐创建失败: " << e.what() << std::endl;}return 0;
}

4.执行结果

四、高级技巧:支持多种预设配置

class MealPreset {
public:static HamburgerMeal createChildrenMeal() {return MealBuilder().setMain("迷你汉堡").setDrink("牛奶").addSide("水果杯").setDessert("小饼干").build();}static HamburgerMeal createComboMeal() {return MealBuilder().setMain("双层牛肉堡").setDrink("中杯可乐").addSide("薯条").build();}
};

五、模式优势深度解析

🚀 C++实现特色优势

  1. 强类型检查:编译期发现类型错误
  2. RAII支持:自动资源管理
  3. 移动语义:高效的对象传递
  4. 灵活内存控制:支持栈对象和智能指针

💡 适用场景扩展

  1. 需要生成的对象有多个变体
  2. 对象创建需要多个步骤的初始化
  3. 需要隔离复杂对象的创建细节
  4. 需要支持不同地区配置(如语言包加载)

六、性能优化策略

  1. 参数预校验:在build()前进行参数检查
  2. 使用移动语义:减少对象拷贝开销
  3. 对象池技术:对频繁创建的对象进行缓存
  4. const正确性:确保构建后的对象不可变

七、与工厂模式对比

特性建造者模式工厂模式
构建重点分步骤构建复杂对象直接创建完整对象
参数处理支持可选参数和分步设置通常需要一次性传递所有参数
对象复杂度适合构建多部件组成的复杂对象适合创建单一结构的对象
扩展性通过新增Builder实现不同配置通过子类化工厂来创建不同对象
典型C++实现链式方法+友元类静态工厂方法/抽象工厂

八、现代C++增强实现

// 使用现代C++特性优化建造者
template<typename T>
class GenericBuilder {
protected:T object;public:operator T() && {  // 右值转换运算符return std::move(object);}T build() && {     // 右值build方法return std::move(object);}
};class ModernMeal : public GenericBuilder<ModernMeal> {
public:ModernMeal& setMain(std::string main) {object.mainItem = std::move(main);return *this;}// 其他设置方法类似...
};

九、最佳实践指南

  1. 防御性编程:在build()中进行参数合法性检查
  2. 清晰的接口设计:保持方法命名直观(withXxx(), addXxx())
  3. 不可变对象:构建完成后锁定对象状态
  4. 文档注释:明确每个构建步骤的作用域和约束条件
  5. 异常安全:确保在异常发生时资源正确释放

总结:构建的艺术

建造者模式如同一位经验丰富的建筑大师,将看似混乱的构建过程转化为标准化的装配流程。在C++的世界中,通过合理运用友元类、移动语义和模板技术,我们能够打造出既高效又灵活的对象构建系统。记住,好的设计模式应用应该像呼吸一样自然,而不是生硬的教条堆砌。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

有哪些免费的SEO软件优化工具

随着2025年互联网的不断发展&#xff0c;越来越多的企业意识到在数字营销中&#xff0c;网站的曝光度和排名至关重要。无论是想要提高品牌知名度&#xff0c;还是想要通过在线销售增加收益&#xff0c;SEO&#xff08;搜索引擎优化&#xff09;都是一项不可忽视的关键策略。而要…

DeepSeek从入门到精通:提示词设计的系统化指南

目录 引言&#xff1a;AIGC时代的核心竞争力 第一部分 基础篇&#xff1a;提示词的本质与核心结构 1.1 什么是提示词&#xff1f; 1.2 提示词的黄金三角结构 第二部分 类型篇&#xff1a;提示词的六大范式 2.1 提示语的本质特征 2.2 提示语的类型 2.2.1 指令型提示词 …

单智能体到多智能体智能体任务规划有什么变化

单智能体到多智能体智能体任务规划有什么变化 核心原理 单智能体任务规划:大模型利用其强大的自然语言理解和生成能力,结合多模态信息,将自然语言描述的任务分解为可执行子任务,并能根据环境反馈调整执行策略。在规划过程中,可通过不同方式生成或优化任务计划,如端到端规…

算法之 跳跃游戏

文章目录 55.跳跃游戏思路参考&#xff1a;56.合并区间 55.跳跃游戏 55.跳跃游戏 灵神思路 思路分析&#xff1a; 两种思路&#xff0c;思路1是我们可以直接维护当前到达i的时候所能到达的最右的边界mr&#xff0c;如果i>mr就说明无法到达i,否则就是可以到达&#xff1b;…

Ubuntu22.04通过Docker部署Jeecgboot

程序发布环境包括docker、mysql、redis、maven、nodejs、npm等。 一、安装docker 1、用如下命令卸载旧Docker: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done 2、安装APT环境依赖包…

STM32 ADC介绍(硬件原理篇)

目录 背景 AD转换器 采样与保持 量化 编码 AD转换器转换原理 DA转换原理 AD转换原理 1.逐次逼近型AD转换器 2.并联比较型AD转换器 编码器 同步D触发器和边沿D触发器 基本RS触发器 同步RS触发器 同步D触发器 边沿型D触发器&#xff08;维持-阻塞D触发器&#xff…

DeepSeek4j 已开源,支持思维链,自定义参数,Spring Boot Starter 轻松集成,快速入门!建议收藏

DeepSeek4j Spring Boot Starter 快速入门 简介 DeepSeek4j 是一个专为 Spring Boot 设计的 AI 能力集成启动器&#xff0c;可快速接入 DeepSeek 大模型服务。通过简洁的配置和易用的 API&#xff0c;开发者可轻松实现对话交互功能。 环境要求 JDK 8Spring Boot 2.7Maven/Gr…

编程技巧:VUE 实现接口返回数据的流式处理

一、写在前面 ChatGPT 的问答响应界面相信大家都见过&#xff0c;内容是一点一点追加式的显示。不是等好了一起发给你&#xff0c;然后一次性展示出来。这种效果和我们平常开发的展示渲染模式有点区别。可能有的同学会说&#xff0c;前端拿到报文后&#xff0c;我们做成这样的…

Django 创建表 choices的妙用:get_<field_name>_display()

1.定义choices 我在创建表时&#xff0c;对于性别这个字段&#xff0c;定义了choices 选项&#xff0c;1代表男&#xff0c;2代表女 mysql中表的数据如下&#xff0c;里面存储的是1或2 如果我们在网页上展示的时候&#xff0c;想展示“男”或“女”&#xff0c;而不是数字1或2…

Mac安装JD-GUI

Mac安装反编译工具步骤如下&#xff1a; 打开官网https://java-decompiler.github.io/ 选择下载mac的安装包解压下载好的压缩包&#xff0c;点击JD-GUI安装 有可能会遇到如下错误。请先检查是否安装JDK&#xff0c;通过java -version命令查看是否是1.8版本的jdk如果jdk没问题&…

[免费]Springboot+Vue医疗(医院)挂号管理系统【论文+源码+SQL脚本】

大家好&#xff0c;我是java1234_小锋老师&#xff0c;看到一个不错的SpringbootVue医疗(医院)挂号管理系统&#xff0c;分享下哈。 项目视频演示 【免费】SpringBootVue医疗(医院)挂号管理系统 Java毕业设计_哔哩哔哩_bilibili 项目介绍 在如今社会上&#xff0c;关于信息上…

链表(典型算法思想)—— OJ例题算法解析思路

目录 一、2. 两数相加 - 力扣&#xff08;LeetCode&#xff09; 算法代码&#xff1a; 1. 初始化 2. 遍历链表并相加 3. 返回结果 举例说明 二、24. 两两交换链表中的节点 - 力扣&#xff08;LeetCode&#xff09; 算法代码&#xff1a; 代码思路 举例说明 初始状…

twisted实现MMORPG 游戏数据库操作封装设计与实现

在设计 MMORPG&#xff08;大规模多人在线角色扮演游戏&#xff09;时&#xff0c;数据库系统是游戏架构中至关重要的一部分。数据库不仅承担了游戏中各种数据&#xff08;如玩家数据、物品数据、游戏世界状态等&#xff09;的存储和管理任务&#xff0c;还必须高效地支持并发访…

lvsDR模式实现

LVS DR模式与NAT模式的优缺点&#xff1a; NAT&#xff1a;优点&#xff1a;配置简单&#xff0c;所需网络环境简单&#xff0c;访问流量的出入都经过LVS服务器&#xff0c;控制流量简单&#xff0c; 缺点&#xff1a;由于访问流量的出入都会经过LVS服务器&#xff0c;所以LVS…

闭源大语言模型的怎么增强:提示工程 检索增强生成 智能体

闭源大语言模型的怎么增强 提示工程 检索增强生成 智能体 核心原理 提示工程:通过设计和优化提示词,引导大语言模型进行上下文学习和分解式思考,激发模型自身的思维和推理能力,使模型更好地理解和生成文本,增强其泛用性和解决问题的能力。检索增强生成:结合检索的准确…

【快速幂算法】快速幂算法讲解及C语言实现(递归实现和非递归实现,附代码)

快速幂算法 快速幂算法可用分治法实现 不难看出&#xff0c;对任意实数a和非负整数n&#xff0c;有&#xff1a; a n { 1 , n 0 , a ≠ 0 0 , a 0 ( a n 2 ) 2 , n > 0 , n 为偶数 ( a n 2 ) 2 ∗ a , n > 0 , n 为奇数 a^n \begin{cases} 1, & n 0, a\neq 0…

大数据SQL调优专题——Hive执行原理

引入 Apache Hive 是基于Hadoop的数据仓库工具&#xff0c;它可以使用SQL来读取、写入和管理存在分布式文件系统中的海量数据。在Hive中&#xff0c;HQL默认转换成MapReduce程序运行到Yarn集群中&#xff0c;大大降低了非Java开发者数据分析的门槛&#xff0c;并且Hive提供命令…

30天开发操作系统 第 20 天 -- API

前言 大家早上好&#xff0c;今天我们继续努力哦。 昨天我们已经实现了应用程序的运行, 今天我们来实现由应用程序对操作系统功能的调用(即API, 也叫系统调用)。 为什么这样的功能称为“系统调用”(system call)呢&#xff1f;因为它是由应用程序来调用(操作)系统中的功能来完…

UE5.2后 Bake Out Materials失效

这个问题出现在5.3&#xff0c;5.4&#xff0c;5.5没有测试 烘焙贴图后会找不到贴图位置&#xff0c; 这个是5.2的正常状态 默认是生成在模型当前目录里&#xff0c;包括新的材质 但是这个bug会让材质和贴图都消失&#xff0c;无法定位 暂时没有办法解决&#xff0c;等官方 …

macOS部署DeepSeek-r1

好奇&#xff0c;跟着网友们的操作试了一下 网上方案很多&#xff0c;主要参考的是这篇 DeepSeek 接入 PyCharm&#xff0c;轻松助力编程_pycharm deepseek-CSDN博客 方案是&#xff1a;PyCharm CodeGPT插件 DeepSeek-r1:1.5b 假设已经安装好了PyCharm PyCharm: the Pyth…