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

news2025/3/24 13:41:00

深入解析建造者模式:用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.coloradmin.cn/o/2300185.html

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

相关文章

uniapp 安卓10+ 选择并上传文件

plus.io.chooseFile({title: 选择文件,filetypes: [mp3], // 允许的文件类型multiple: false, // 是否允许多选}, (res) > {console.log(虚拟路径666&#xff1a;, res);var arr[{name: files,uri: res.files[0],}]let obj {"tableName": "mingmen_daily_mi…

【第1章:深度学习概览——1.6 深度学习框架简介与选择建议】

嘿,各位老铁们,今天咱们来一场深度学习框架的深度探索之旅。在这个充满无限可能的深度学习时代,深度学习框架就像是连接理论与实践的桥梁,帮助我们从算法设计走向实际应用。随着技术的飞速发展,深度学习框架的选择变得越来越多样化,每一种框架都有其独特的优势和适用场景…

网页制作02-html,css,javascript初认识のhtml的文字与段落标记

用一首李白的将进酒,对文字与段落标记进行一个简单的介绍演示&#xff1a; 目录 一、标题字 1、标题字标记h 2、标题字对其属性align 二、文本基本标记 1、字体属性face 2、字号属性size 3、颜色属性 Color 三、文本格式化标记 1、粗体标记 b &#xff0c;strong 2、…

一.数据治理理论架构

1、数据治理核心思想&#xff1a; 数据治理理论架构图描绘了一个由顶层设计、管控机制、核心领域和管理系统四个主要部分组成的数据治理框架。它旨在通过系统化的方法&#xff0c;解决数据治理机制缺失引发的业务和技术问题&#xff0c;并最终提升企业的数据管理水平。 数据治…

PHP基础部分

但凡是和输入、写入相关的一定要预防别人植入恶意代码! HTML部分 语句格式 <br> <hr> 分割符 <p>插入一行 按住shift 输入! 然后按回车可快速输入html代码(VsCode需要先安装live server插件) html:<h1>标题 数字越大越往后</h1> <p…

人工智能 - 主动视觉可能就是你所需要的:在双臂机器人操作中探索主动视觉

AV-ALOHA 系统使用用于 AV 的 VR 耳机实现直观的数据收集&#xff0c;并且 用于作的 VR 控制器或引线臂。这有助于捕捉全身和头部 远程作我们的真实和模拟系统的运动&#xff0c;记录来自 6 个的视频 不同的摄像头&#xff0c;并为我们的 AV 仿制学习策略提供训练数据。 加州大…

DeepSeek 助力 Vue 开发:打造丝滑的日期选择器(Date Picker),未使用第三方插件

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

Kafka偏移量管理全攻略:从基础概念到高级操作实战

#作者&#xff1a;猎人 文章目录 前言&#xff1a;概念剖析kafka的两种位移消费位移消息的位移位移的提交自动提交手动提交 1、使用--to-earliest重置消费组消费指定topic进度2、使用--to-offset重置消费offset3、使用--to-datetime策略指定时间重置offset4、使用--to-current…

一周学会Flask3 Python Web开发-Debug模式开启

锋哥原创的Flask3 Python Web开发 Flask3视频教程&#xff1a; 2025版 Flask3 Python web开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili 默认情况&#xff0c;项目开发是普通模式&#xff0c;也就是你修改了代码&#xff0c;必须重启项目&#xff0c;新代码才生效&…

java练习(28)

ps&#xff1a;练习来自力扣 给定一个二叉树&#xff0c;判断它是否是平衡二叉树 // 定义二叉树节点类 class TreeNode {int val;TreeNode left;TreeNode right;TreeNode() {}TreeNode(int val) { this.val val; }TreeNode(int val, TreeNode left, TreeNode right) {this.va…

Web安全|渗透测试|网络安全

基础入门(P1-P5) p1概念名词 1.1域名 什么是域名&#xff1f; 域名&#xff1a;是由一串用点分隔的名字组成的Internet上某一台计算机或计算机组的名称&#xff0c;用于在数据传输时对计算机的定位标识&#xff08;有时也指地理位置&#xff09;。 什么是二级域名多级域名&am…

OpenHarmony 系统性能优化——默认关闭全局动画

笔者最近发现&#xff0c;关闭OpenHarmony全局动画&#xff0c;系统UI的响应速度会极大的提升 1.全局动画的开关由系统属性persist.sys.arkui.animationscale来控制&#xff0c;默认为1。也就是 动画缩放 1x 2.如果让persist.sys.arkui.animationscale默认为0,也就是关闭的状态…

C 程序多线程拆分文件

C 程序多线程拆分文件 在C语言中&#xff0c;实现多线程来拆分文件通常需要借助多线程库&#xff0c;比如 POSIX 线程库&#xff08;pthread&#xff09;或者 Windows 的线程库&#xff08;CreateThread 或类似的函数&#xff09;。下面我将分别展示在 Linux 和 Windows 环境下…

【Linux】Ubuntu Linux 系统——Python集成开发环境

ℹ️大家好&#xff0c;我是练小杰&#xff0c;今天周四了&#xff0c;明天就周五了&#xff0c;再坚持坚持又能休息了&#xff01;&#xff01;&#x1f606; 本文是有关Linux 操作系统中Python集成开发环境基础知识&#xff0c;后续将添加更多相关知识噢&#xff0c;谢谢各位…

数据库加密全解析:从传输到存储的安全实践

title: 数据库加密全解析:从传输到存储的安全实践 date: 2025/2/17 updated: 2025/2/17 author: cmdragon excerpt: 数据加密是数据库安全的最后一道物理防线。传输层SSL/TLS配置、存储加密技术及加密函数实战应用,覆盖MySQL、PostgreSQL、Oracle等主流数据库的20+生产级加密…

【Prometheus】prometheus结合domain_exporter实现域名监控

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全…

计算机专业知识【软件开发中的常用图表:E - R图、HIPO、DFD、N - S、PAD详解】

在软件开发过程中&#xff0c;有许多种图表工具被用于不同阶段的设计和分析&#xff0c;帮助开发者更清晰地理解系统结构、数据流程和算法逻辑。下面将详细介绍E - R图、HIPO图、DFD图、N - S图和PAD图&#xff0c;包括它们的样子和用途。 一、E - R图&#xff08;实体 - 联系…

智能马达保护器:为工业电机安全运行保驾护航

在工业生产中&#xff0c;电动机作为核心动力设备&#xff0c;其稳定运行直接关系到生产效率与安全性。然而&#xff0c;复杂的工况环境、频繁启停和突发负载变化&#xff0c;常导致电机面临过载、缺相、短路等故障风险。安科瑞智能马达保护器凭借其智能化、高精度、多功能的设…

深度集成DeepSeek大模型:WebSocket流式聊天实现

目录 5分钟快速接入DeepSeek大模型&#xff1a;WebSocket实时聊天指南创建应用开发后端代码 (Python/Node.js)结语 5分钟快速接入DeepSeek大模型&#xff1a;WebSocket实时聊天指南 创建应用 访问DeepSeek官网 前往 DeepSeek官网。如果还没有账号&#xff0c;需要先注册一个。…

QT基础二、信号和槽

一、什么是信号和槽&#xff1f; 1、简述 在Qt框架中&#xff0c;信号和槽&#xff08;Signals and Slots&#xff09; 是一种用于对象间通信的机制。它是一种非常强大且灵活的设计模式&#xff0c;广泛应用于事件驱动编程中。信号和槽机制允许对象之间以松耦合的方式进行交互…