(五)趣学设计模式 之 建造者模式!

news2025/2/24 10:21:28

在这里插入图片描述

目录

    • 一、 啥是建造者模式?
    • 二、 为什么要用建造者模式?
    • 三、 建造者模式怎么实现?
    • 四、 建造者模式的应用场景
    • 五、 建造者模式的优点和缺点
    • 六、 总结

🌟我的其他文章也讲解的比较有趣😁,如果喜欢博主的讲解方式,可以多多支持一下,感谢🤗!
🌟了解原型模式请看: (四)趣学设计模式 之 原型模式!

这篇文章带你详细认识一下设计模式中的建造者模式

一、 啥是建造者模式?

建造者模式,就像组装乐高玩具 🧸! 你有一堆零件,想要拼出一个复杂的模型,比如一辆汽车 🚗 或者一栋房子 🏠,但是直接拼太麻烦了,容易出错,而且不同的汽车和房子,拼装步骤可能不一样。 建造者模式就是把组装过程分解成一步一步的,每一步负责一部分,比如安装引擎、安装轮胎、刷油漆等等,每一步都由专门的工人(具体建造者)来完成,最后把所有部分组合起来,就得到了完整的模型!

  • 对象创建过程复杂: 就像盖房子 🏠,需要打地基、砌墙、安装门窗、装修等等,一步都不能少,而且顺序也很重要!
  • 需要灵活控制对象创建过程: 就像定制汽车 🚗,可以根据自己的喜好选择不同的引擎、轮胎、颜色等等,每个配置都会影响最终的汽车!
  • 希望隐藏对象内部的构建细节: 就像吃汉堡 🍔,你只想吃,不想知道它是怎么做的,更不想知道厨师是怎么一步一步把汉堡做出来的!

二、 为什么要用建造者模式?

用建造者模式,好处多多:

  • 分工明确: 每个建造者负责一部分,职责清晰,代码更易维护! 就像盖房子,有专门的泥瓦匠砌墙,有专门的木匠安装门窗,每个人都只负责自己的部分,不会互相干扰!
  • 灵活定制: 可以根据需要选择不同的建造者,创建不同配置的对象! 就像定制汽车,可以选择不同的引擎、轮胎、颜色等等,最终得到一辆完全符合自己需求的汽车!
  • 隐藏细节: 客户端不用关心对象的构建过程,只需要知道最终的结果! 就像吃汉堡,你只需要知道汉堡很好吃,不用关心厨师是怎么一步一步把汉堡做出来的!
  • 代码复用: 相同的构建步骤可以被不同的建造者复用! 就像盖房子,不同的房子可能都需要打地基,这个步骤就可以被不同的建造者复用!

三、 建造者模式怎么实现?

建造者模式主要包含以下几个角色:

  • 产品(Product): 最终要创建的对象,就像乐高模型、汽车、房子等等!
  • 抽象建造者(Builder): 定义了构建产品的接口,就像乐高说明书、汽车设计图纸、房屋设计图纸等等! 它规定了构建产品的各个步骤,但是具体的实现由具体建造者来完成!
  • 具体建造者(ConcreteBuilder): 实现了抽象建造者的接口,负责构建产品的各个部分,就像乐高工人、汽车工人、建筑工人等等! 他们按照抽象建造者的规定,一步一步地构建产品!
  • 指挥者(Director): 负责安排构建的顺序,就像乐高设计师、汽车设计师、房屋设计师等等! 他们知道构建产品的正确顺序,并指挥具体建造者按照这个顺序来构建产品!

代码示例:

// 1. 产品类 (Product)
class Computer {
    private String cpu; // CPU
    private String ram; // 内存
    private String hardDisk; // 硬盘
    private String graphicsCard; // 显卡

    // 设置 CPU
    public void setCpu(String cpu) {
        this.cpu = cpu;
    }

    // 设置 内存
    public void setRam(String ram) {
        this.ram = ram;
    }

    // 设置 硬盘
    public void setHardDisk(String hardDisk) {
        this.hardDisk = hardDisk;
    }

    // 设置 显卡
    public void setGraphicsCard(String graphicsCard) {
        this.graphicsCard = graphicsCard;
    }

    // 重写 toString 方法,方便打印电脑信息
    @Override
    public String toString() {
        return "Computer{" +
                "cpu='" + cpu + '\'' +
                ", ram='" + ram + '\'' +
                ", hardDisk='" + hardDisk + '\'' +
                ", graphicsCard='" + graphicsCard + '\'' +
                '}';
    }
}

// 2. 抽象建造者 (Builder)
interface ComputerBuilder {
    void buildCpu(String cpu); // 构建 CPU
    void buildRam(String ram); // 构建 内存
    void buildHardDisk(String hardDisk); // 构建 硬盘
    void buildGraphicsCard(String graphicsCard); // 构建 显卡
    Computer build(); // 返回最终产品
}

// 3. 具体建造者 (ConcreteBuilder)
class GamingComputerBuilder implements ComputerBuilder {
    private Computer computer = new Computer(); // 创建一个电脑对象

    // 构建 CPU
    @Override
    public void buildCpu(String cpu) {
        computer.setCpu(cpu);
    }

    // 构建 内存
    @Override
    public void buildRam(String ram) {
        computer.setRam(ram);
    }

    // 构建 硬盘
    @Override
    public void buildHardDisk(String hardDisk) {
        computer.setHardDisk(hardDisk);
    }

    // 构建 显卡
    @Override
    public void buildGraphicsCard(String graphicsCard) {
        computer.setGraphicsCard(graphicsCard);
    }

    // 返回最终产品
    @Override
    public Computer build() {
        return computer;
    }
}

// 4. 指挥者 (Director)
class ComputerDirector {
    private ComputerBuilder builder; // 依赖一个 ComputerBuilder

    // 构造函数,传入一个 ComputerBuilder
    public ComputerDirector(ComputerBuilder builder) {
        this.builder = builder;
    }

    // 构建电脑
    public Computer construct(String cpu, String ram, String hardDisk, String graphicsCard) {
        builder.buildCpu(cpu); // 构建 CPU
        builder.buildRam(ram); // 构建 内存
        builder.buildHardDisk(hardDisk); // 构建 硬盘
        builder.buildGraphicsCard(graphicsCard); // 构建 显卡
        return builder.build(); // 返回最终产品
    }
}

// 5. 客户端使用
public class Client {
    public static void main(String[] args) {
        // 创建具体建造者
        GamingComputerBuilder gamingComputerBuilder = new GamingComputerBuilder();

        // 创建指挥者
        ComputerDirector computerDirector = new ComputerDirector(gamingComputerBuilder);

        // 构建电脑
        Computer gamingComputer = computerDirector.construct("Intel i9", "32GB", "1TB SSD", "Nvidia RTX 3080");

        // 打印电脑信息
        System.out.println("游戏电脑: " + gamingComputer);
    }
}

代码解释:

  • Computer:产品类,表示电脑。 就像乐高模型,是最终要组装完成的东西!
  • ComputerBuilder:抽象建造者,定义了构建电脑的接口。 就像乐高说明书,规定了组装电脑的各个步骤!
  • GamingComputerBuilder:具体建造者,实现了 ComputerBuilder 接口,负责构建游戏电脑。 就像乐高工人,按照说明书一步一步地组装电脑!
  • ComputerDirector:指挥者,负责安排构建的顺序。 就像乐高设计师,知道组装电脑的正确顺序,并指挥工人按照这个顺序来组装电脑!

输出结果:

游戏电脑: Computer{cpu='Intel i9', ram='32GB', hardDisk='1TB SSD', graphicsCard='Nvidia RTX 3080'}

分析:

客户端只需要告诉指挥者需要什么配置的电脑,指挥者会安排具体建造者一步一步地构建电脑,最后返回完整的电脑对象。 客户端不需要关心电脑是怎么一步一步组装起来的,只需要知道最终的结果!

四、 建造者模式的应用场景

  • 创建复杂的对象: 就像创建汽车 🚗、房子 🏠、飞机 ✈️ 等等,需要多个步骤才能完成,而且每个步骤都很复杂!
  • 需要灵活控制对象创建过程: 就像定制家具 🪑,可以根据自己的喜好选择不同的材料、颜色和尺寸,最终得到一件完全符合自己需求的家具!
  • 需要隐藏对象内部的构建细节: 就像制作蛋糕 🎂,你只想吃,不想知道它是怎么做的,更不想知道厨师是怎么一步一步把蛋糕做出来的!
  • 需要创建不同表示的对象: 就像创建不同风格的房子 🏘️,可以使用不同的建造者来创建,比如现代风格的房子、古典风格的房子等等!

五、 建造者模式的优点和缺点

优点:

  • 分工明确: 每个建造者负责一部分,职责清晰,代码更易维护! 就像一个团队,每个人都只负责自己的部分,不会互相干扰,提高了工作效率!
  • 灵活定制: 可以根据需要选择不同的建造者,创建不同配置的对象! 就像定制汽车,可以选择不同的引擎、轮胎、颜色等等,最终得到一辆完全符合自己需求的汽车!
  • 隐藏细节: 客户端不用关心对象的构建过程,只需要知道最终的结果! 就像吃汉堡,你只需要知道汉堡很好吃,不用关心厨师是怎么一步一步把汉堡做出来的!
  • 代码复用: 相同的构建步骤可以被不同的建造者复用! 就像盖房子,不同的房子可能都需要打地基,这个步骤就可以被不同的建造者复用,减少了代码的重复!
  • 易于扩展: 可以很容易地添加新的建造者,来创建新的对象类型! 就像乐高玩具,可以不断推出新的模型,只需要添加新的乐高说明书和乐高工人即可!

缺点:

  • 代码复杂: 需要创建多个类,代码量比较大! 就像一个团队,需要更多的人员,增加了管理的复杂性!
  • 抽象性高: 理解起来比较抽象,需要一定的设计经验! 就像学习乐高说明书,需要一定的空间想象能力!
  • 修改困难: 如果需要修改产品的内部结构,可能需要修改多个类! 就像修改乐高模型,可能需要修改乐高说明书和乐高零件!

六、 总结

  • 建造者模式就像组装乐高玩具,把复杂的对象创建过程分解成一步一步的!
  • 适用于对象创建过程复杂、需要灵活控制对象创建过程、希望隐藏对象内部的构建细节的场景!
  • 主要包含产品、抽象建造者、具体建造者和指挥者四个角色!
  • 优点是分工明确、灵活定制、隐藏细节、代码复用、易于扩展!
  • 缺点是代码复杂、抽象性高、修改困难!

希望这篇文章能让你彻底理解建造者模式! 👍

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

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

相关文章

github 怎么创建一个私有repository 并从另外一台电脑拉取下来更新

1.github上新建一个repository 设置为private tips删除在这 点setting 然后往下拖动 会有个这里是用来删项目的 2.另外 一台电脑拉取这个repository的时候 需要配置 一个ssh key 这个key的内容生成参考本地电脑的生成 然后在这配置 2.1 生成 SSH 密钥(如果还没有…

DeepSeek-R1本地化部署的硬件要求

DeepSeek-R1本地化部署的硬件要求全解析 引言 DeepSeek-R1作为一款高效的AI推理模型,凭借其卓越的推理性能和灵活的训练机制,成为了春节期间的热议话题。 然而,要在本地成功部署DeepSeek-R1,尤其是其满载的 671B 参数版本&#…

AGI觉醒假说的科学反驳:从数学根基到现实约束的深度解析

文章目录 引言:AGI觉醒论的核心迷思一、信息论视角:意识产生的熵约束1.1 香农熵的物理极限1.2 量子退相干的时间屏障二、数学根基:形式系统的自指困境2.1 哥德尔不完备定理的现代诠释三、概念解构:AGI觉醒假说的认知陷阱3.1 术语混淆的迷雾3.2 拟人化谬误的认知根源四、意识…

CSS—盒模型(3分钟结合示例精通盒模型)

个人博客:haichenyi.com。感谢关注 1. 目录 1–目录2–概念3–内容4–内边距5–边框6–外边距7–类型 概念 在HTML中,每一个元素都可以看作一个矩形的盒子。如图 如上图所示,一个一个的矩形都可以堪称一个元素。矩形有大有小,边有…

蓝桥杯 3.搜索

蓝桥杯 3.搜索 文章目录 蓝桥杯 3.搜索DFS回溯DFS剪枝记忆化搜索编程66-75 DFS回溯 回溯法简介 使用**DFS(深度优先搜索)**实现, DFS是一种遍历或搜索图, 树或者图像等数据结构的算法, 当然这个图, 树未必要存储下来(隐式处理就是回溯法)搜索树一般是排列型搜索树 (总节点个数…

STM32的“Unique device ID“能否修改?

STM32F1系列的"Unique device ID"寄存器的地址为0x1FFFF7E8。 这个寄存器是只读的。 "Unique device ID"寄存器位于“System memory”中。“System memory”地址范围为“0x1FFF F000- 0x1FFF F7FF”。 所有STM32 MCU上都存在系统引导加载程序。顾名思义&a…

[内网基础] 内网基础知识 —— Windows 工作组

关注这个专栏的其他相关笔记:[内网安全] 内网渗透 - 学习手册-CSDN博客 0x01:Windows 工作组介绍 在一个大型单位里,可能有成百上千台计算机互相连接组成局域网,如果不对这些计算机进行分组,网络的混乱程度是可想而知…

【新手初学】SQL注入之二次注入、中转注入、伪静态注入

二次注入 一、概念 二次注入可以理解为,攻击者构造的恶意数据存储在数据库后,恶意数据被读取并进入到SQL查询语句所导致的注入。 二、原理 防御者可能在用户输入恶意数据时对其中的特殊字符进行了转义处理,但在恶意数据插入到数据库时被处…

Deepseek存算分离安全部署手册

Deepseek大火后,很多文章教大家部署Dfiy和ollamadeepseek,但是大部分都忽略了数据安全问题,本文重点介绍Deepseek存算分裂安全架设,GPU云主机只负责计算、CPU本地主机负责数据存储,确保数据不上云,保证私有…

单页图床HTML源码+本地API接口图床系统修复版源码

源码介绍 图床系统是一种用于存储和管理图片文件的在线服务。它允许用户上传图片文件,并生成相应的图片链接,从而方便用户在网页、社交媒体或其他平台上分享图片。 PS:源码压缩包分为两个版本,一个是调用360第三方api接口,另外一…

IDEA使用Maven方式构建SpringBoot项目

1、环境准备 确保你已经安装了以下工具: Java JDK(推荐 JDK 8 或更高版本) IntelliJ IDEA(推荐使用最新版本) 2、创建 Spring Boot 项目 (1) 打开 IntelliJ IDEA。 (2&#xff09…

【SPIE出版,见刊快速,EI检索稳定,浙江水利水电学院主办】2025年物理学与量子计算国际学术会议(ICPQC 2025)

2025年物理学与量子计算国际学术会议(ICPQC 2025)将于2025年4月18-20日在中国杭州举行。本次会议旨在汇聚全球的研究人员、学者和业界专家,共同探讨物理学与量子计算领域的最新进展与前沿挑战。随着量子技术的快速发展,其在信息处…

查看cmd下python的安装路径 + Windows 命令行添加环境变量和不重启刷新环境变量

1、查看cmd下python的安装路径 cmd ----》输入“python” 命令 ---》输入 import sys; print(sys.executable) 即可看到当前系统python的安装路径: 注:系统所使用的python实际上就是在系统环境变量下配置的python目录。 2、刷新path命令:在c…

C/C++跳动的爱心

系列文章 序号直达链接1C/C李峋同款跳动的爱心2C/C跳动的爱心3C/C经典爱心4C/C满屏飘字5C/C大雪纷飞6C/C炫酷烟花7C/C黑客帝国同款字母雨8C/C樱花树9C/C奥特曼10C/C精美圣诞树11C/C俄罗斯方块小游戏12C/C贪吃蛇小游戏13C/C孤单又灿烂的神14C/C闪烁的爱心15C/C哆啦A梦16C/C简单…

Cannot deserialize instance of java.lang.String out of START_ARRAY token

这个错误 Cannot deserialize instance of java.lang.String out of START_ARRAY token 表示 Jackson 正在尝试将一个 JSON 数组反序列化成一个 String 类型的字段,但是 JSON 中传递的是一个数组而不是单一的字符串。 具体来说,这段堆栈信息&#xff1a…

一、初始爬虫

1.爬虫的相关概念 1.1 什么是爬虫 网络爬虫(又被称为网页蜘蛛,网络机器人)就是模拟浏览器发送网络请求,接收请求响应,一种按照一定的规则,自动地爬取互联网信息的程序。 原则上,只要是浏览器…

在低功耗MCU上实现人工智能和机器学习

作者:Silicon Labs 人工智能(AI)和机器学习(ML)技术不仅正在快速发展,还逐渐被创新性地应用于低功耗的微控制器(MCU)中,从而实现边缘AI/ML解决方案。这些MCU是许多嵌入式…

QQ登录测试用例报告

QQ登录测试用例思维导图 一、安全性测试用例 1. 加密传输与存储验证 测试场景:输入账号密码并提交登录请求。预期结果:账号密码通过加密传输(如HTTPS)与存储(如哈希加盐),无明文暴露。 2. 二…

细说STM32F407单片机2个ADC使用DMA同步采集各自的1个输入通道的方法

目录 一、示例说明 二、工程配置 1、RCC、DEBUG、CodeGenerator 2、USART6 3、TIM3 (1)Mode (2)参数设置 (3) TRGO (4)ADC1_IN0 1)ADCs_Common_Settings 2&a…

[python脚本]论文1.(一)CPU/内存数据分析和分组

CPU 收集到的CPU数据,格式如下: 由于这里6个数据为一组来收集latency的数据以及各个分位值的数据,而本质上每一行都是一次完整的测试,因此这里将这个csv文件分为两个文件,第一个是和latency相关的,将6条数…