设计模式 创建型 建造者模式(Builder Pattern)与 常见技术框架应用 解析
建造者模式,又称生成器模式,是一种对象构建模式。它主要用于构建复杂对象,通过将复杂对象的构建过程与其表示分离,使得同样的构建过程可以创建出具有不同表示的对象。该模式的核心思想是将一个复杂对象的构建过程分解为多个简单的步骤,并允许用户通过指定复杂对象的类型和内容来构建它们,而无需知道对象内部的具体构建细节。
一、核心思想
建造者模式的核心思想是将复杂对象的构建过程与它的表示分离,使得构建过程可以灵活地组合,而不会影响最终的产品表示。这有助于创建一个逐步构建的复杂对象,同时隐藏其内部构造细节。
二、定义与结构
- 定义:建造者模式是一种创建型设计模式,它允许将一个复杂对象的构建与其表示分离,使得同样的构建过程可以创建不同的表示。通过将对象的构建过程分解为一系列步骤,由不同的建造者类来实现这些步骤,最后由指挥者类来统筹构建流程,以得到最终的复杂对象。
- 结构:
- 产品(Product):即要构建的复杂对象,它具有多个组成部分,这些部分通常由建造者逐步构建完成。例如上述的定制电脑,它就是产品,包含 CPU、内存等各种硬件组件作为其内部属性。
- 抽象建造者(Abstract Builder):定义了构建产品各个部分以及返回最终产品的抽象方法,为具体建造者提供统一的接口规范。比如定义了安装 CPU、安装内存等抽象操作方法。
- 具体建造者(Concrete Builder):实现抽象建造者接口,负责按照特定的需求实际构建产品的各个部分,并在最后返回构建完成的产品。例如,高性能电脑建造者会选择高端的 CPU、大容量内存来组装电脑。
- 指挥者(Director):负责指挥和协调建造者的构建步骤,它知晓构建产品所需的完整流程,通过调用建造者的方法来实现产品的构建,但它并不负责具体的构建细节,只是掌控构建的节奏。比如指挥者决定先安装哪个部件,后安装哪个部件。
角色
- 产品角色(Product):作为建造的目标对象,它承载了复杂对象的所有属性和状态,是建造者们努力打造的成果,通常是一个包含多个成员变量的类,这些变量代表了对象的不同组成部分。
- 抽象建造者角色(Abstract Builder):作为建造者的抽象模板,规定了构建产品必须实现的一组抽象方法,这些方法涵盖了从基础部件安装到最终产品组装成型的各个环节,为具体建造者的实现提供了方向指引。
- 具体建造者角色(Concrete Builder):依据抽象建造者的蓝图,深入到实际构建工作的第一线,根据特定的设计要求或用户需求,精心挑选合适的部件,运用专业的技术,将一个个零散的部件逐步组装成完整的产品,并且能够将最终成果交付给客户端。
- 指挥者角色(Director):站在更高的层面统筹规划,它熟悉整个构建流程的最优顺序,通过向具体建造者下达精准的指令,协调各个建造者之间的工作衔接,确保复杂对象能够高效、有序地构建完成,虽然不直接参与物理构建,但却掌控着构建的全局节奏。
三、实现步骤及代码示例(以构建一个网页表单为例)
首先,定义产品:
// 产品:网页表单
public class WebForm {private String formTitle;private String formFields;private String submitButtonText;public WebForm(String formTitle, String formFields, String submitButtonText) {this.formTitle = formTitle;this.formFields = formFields;this.submitButtonText = submitButtonText;}// 省略 getters 和 setters,方便展示主要逻辑@Overridepublic String toString() {return "WebForm{" +"formTitle='" + formTitle + '\'' +"formFields='" + formFields + '\'' +"submitButtonText='" + submitButtonText + '\'' +'}';}
}
接着,定义抽象建造者:
// 抽象建造者:网页表单建造者
public abstract class WebFormBuilder {protected WebForm webForm;public WebForm getWebForm() {return webForm;}public abstract void buildFormTitle();public abstract void buildFormFields();public abstract void buildSubmitButtonText();
}
然后,创建具体建造者:
// 具体建造者:登录表单建造者
public class LoginFormBuilder extends WebFormBuilder {@Overridepublic void buildFormTitle() {webForm = new WebForm("Login Form", "", "");}@Overridepublic void buildFormFields() {webForm.setFormFields("Username: <input type='text' name='username'><br>Password: <input type='password' name='password'>");}@Overridepublic void buildSubmitButtonText() {webForm.setSubmitButtonText("Login");}
}// 具体建造者:注册表单建造者
public class RegistrationFormBuilder extends WebFormBuilder {@Overridepublic void buildFormTitle() {webForm = new WebForm("Registration Form", "", "");}@Overridepublic void buildFormFields() {webForm.setFormFields("Username: <input type='text' name='username'><br>Password: <input type='password' name='password'><br>Email: <input type='email' name='email'>");}@Overridepublic void buildSubmitButtonText() {webForm.setSubmitButtonText("Register");}
}
再定义指挥者:
// 指挥者:表单指挥者
public class FormDirector {private WebFormBuilder webFormBuilder;public FormDirector(WebFormBuilder webFormBuilder) {this.webFormBuilder = webFormBuilder;}public WebForm construct() {webFormBuilder.buildFormTitle();webFormBuilder.buildFormFields();webFormBuilder.buildSubmitButtonText();return webFormBuilder.getWebForm();}
}
最后,在客户端代码中使用:
public class Main {public static void main(String[] args) {// 构建登录表单LoginFormBuilder loginFormBuilder = new LoginFormBuilder();FormDirector formDirector = new FormDirector(loginFormBuilder);WebForm loginForm = formDirector.construct();System.out.println(loginForm);// 构建注册表单RegistrationFormBuilder registrationFormBuilder = new RegistrationFormBuilder();formDirector = new FormDirector(registrationFormBuilder);WebForm registrationForm = formDirector.construct();System.out.println(registrationForm);}
}
四、常见技术框架应用代码示例
建造者模式(Builder Pattern)是一种创建型设计模式,它通过将复杂对象的构建过程与其表示分离,使得同样的构建过程可以创建不同的表示。这种模式的优点在于能够分步骤构建复杂对象,而无需关注这些部分的具体组装过程,从而提高了代码的灵活性和可扩展性。关于建造者模式在技术框架中的应用,虽然具体的应用场景可能因框架而异,但以下是一些常见的技术框架和领域中建造者模式的应用示例:
-
Java EE和Spring框架:
- 在Java EE和Spring等企业级应用框架中,建造者模式常用于构建具有复杂属性和依赖关系的对象。例如,在构建数据库连接池、消息队列连接等复杂对象时,可以使用建造者模式来逐步设置对象的属性,并最终返回一个配置好的实例。
-
Web开发框架:
- 在Web开发中,建造者模式常用于构建HTTP请求和响应对象。这些对象通常包含多个属性和配置选项,如URL、请求方法、头信息、请求体等。通过建造者模式,可以逐步设置这些属性,而不是在构造函数中一次性传递所有参数。
-
图形用户界面(GUI)框架:
- 在构建复杂的GUI组件时,如窗口、按钮、文本框等,建造者模式可以帮助开发者逐步设置组件的属性,如大小、位置、颜色等。这样不仅可以提高代码的可读性,还可以使组件的创建过程更加灵活和可扩展。
-
游戏开发框架:
- 在游戏开发中,建造者模式常用于构建游戏中的角色、道具、场景等复杂对象。通过建造者模式,可以逐步设置这些对象的属性,如生命值、攻击力、防御力等,并最终生成一个完整的游戏对象。
-
配置和初始化框架:
- 在一些配置和初始化框架中,建造者模式常用于构建配置对象。这些配置对象通常包含多个配置项和依赖关系,通过建造者模式可以逐步设置这些配置项,并最终生成一个配置好的实例。
在 Android 开发中,AlertDialog
的创建就运用了类似建造者模式的思想。
import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;public class DialogBuilderExample {public static void main(String[] args) {Context context = null; // 实际开发中需替换为有效的上下文AlertDialog.Builder builder = new AlertDialog.Builder(context);builder.setTitle("提示").setMessage("这是一个示例对话框").setPositiveButton("确定", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {// 点击确定按钮的处理逻辑}}).setNegativeButton("取消", new DialogInterface.OnClickListener() {@@Overridepublic void onClick(DialogInterface dialog, int which) {// 点击取消按钮的处理逻辑}});AlertDialog dialog = builder.create();dialog.show();}
}
这里,AlertDialog.Builder
相当于具体建造者,它逐步构建 AlertDialog
(产品)的各个部分,如标题、消息内容、按钮等,而我们不需要关心 AlertDialog
内部复杂的创建和初始化过程,只需按照需求配置各个部件即可,最后通过 create
方法得到完整的对话框并展示。
五、应用场景
- 复杂对象构建:当需要创建的对象具有大量的属性、复杂的初始化逻辑,并且不同的属性组合会产生不同的功能或外观变体时,建造者模式非常适用。比如构建一辆汽车,涉及发动机、底盘、车身、内饰等多个复杂部件的选型与组装,使用建造者模式可以清晰地划分构建步骤,方便管理。
- 对象构建过程需要精细控制:如果对对象构建的顺序、步骤有严格要求,建造者模式能确保构建过程按照预定的流程进行。例如在建筑施工中,先打地基、再建主体、最后装修,通过指挥者精确安排建造者的工作顺序,保证建筑质量。
- 创建对象的逻辑频繁变化:当创建对象的逻辑需要经常调整,比如根据不同的市场需求,产品的配置、样式经常改变,建造者模式将构建逻辑封装在建造者和指挥者类中,修改时只需在相关类中进行,不影响客户端使用。
六、优缺点
- 优点:
- 构建过程与表示分离:使得客户端不需要了解复杂对象的具体构建细节,降低了耦合度,客户端只需与指挥者和建造者的抽象接口交互,即使构建过程变化,也不影响客户端代码。
- 精细控制构建过程:通过指挥者可以精确安排构建步骤,满足不同的构建需求,确保复杂对象按照预期的方式构建,提高构建质量。
- 代码复用性高:抽象建造者和具体建造者类可以在多个地方复用,尤其是当有多种类似的复杂对象需要构建时,复用建造者类能节省开发时间。例如,不同风格的房屋建造可以复用基础的建造者类。
- 便于维护和扩展:当需要增加新的构建步骤、修改构建逻辑或创建新的复杂对象变体时,只需在建造者和指挥者类中操作,不会影响其他部分代码。如要给汽车增加自动驾驶功能模块,只需在相关建造者类中添加构建步骤。
- 缺点:
- 增加代码复杂性:引入建造者模式会增加类的数量,包括抽象建造者、具体建造者、指挥者以及产品类,对于简单的对象创建场景,会使代码显得过于繁琐。
- 对开发人员要求较高:要求开发人员熟悉建造者模式的结构和原理,在设计和实现过程中需要合理规划各个角色的职责和交互,否则容易导致代码混乱,增加维护难度。
建造者模式为创建复杂对象提供了一种高效、灵活且易于维护的解决方案,虽然在简单场景下可能略显繁琐,但在应对复杂构建需求时能充分发挥其优势,提升软件的质量和开发效率。
相关文章:

设计模式 创建型 建造者模式(Builder Pattern)与 常见技术框架应用 解析
建造者模式,又称生成器模式,是一种对象构建模式。它主要用于构建复杂对象,通过将复杂对象的构建过程与其表示分离,使得同样的构建过程可以创建出具有不同表示的对象。该模式的核心思想是将一个复杂对象的构建过程分解为多个简单的…...

嵌入式系统中C++的基本使用方法
大家好,今天主要给大家分享一下,最近操作C++代码的控制方法。 什么是构造函数?构造函数在对象实例化时被系统自动调用,仅且调用一次。 什么是析构函数?与构造函数相反, 在对象结束其生命周期时系统自动执行析构函数。 第一个:析构函数与构造函数区别 实例代码: #inclu…...

机器人C++开源库The Robotics Library (RL)使用手册(四)
建立自己的机器人3D模型和运动学模型 这里以国产机器人天机TR8为例,使用最普遍的DH运动学模型,结合RL所需的描述文件,进行生成。 最终,需要的有两个文件,一个是.wrl三维模型描述文件;一个是.xml运动学模型描述文件。 1、通过STEP/STP三维文件生成wrl三维文件 机器人的…...

在 uni-app 中使用 wxml-to-canvas 的踩坑经验总结
在 uni-app 中使用 wxml-to-canvas 的踩坑经验总结 wxml-to-canvas 是一款非常强大的小程序工具,可以将 WXML 转换为 Canvas 绘图,用于生成海报、分享图片等。将其应用于 uni-app 项目中,可以为多端开发带来极大的便利,但也有一些…...

视频智能翻译
i68,爱六八,链接你我他 EasyVideoTrans英文视频转换成中文视频 EasyVideoTrans简要 最快的英文视频转中文方案由B站多位程序员Up主共同协作开发开源的项目在线Demo:EasyVideoTrans前端项目:https://github.com/sutro-planet/easyvideotrans-frontend后端项目:https://github…...

《Python加解密小实验:探索数据加密与解密的世界》
铺垫(1)-源码 import hashlib source "你好" # print(hashlib.md5(source.encode()).hexdigest())# 文件加解密 with open(../文件引用/index.png, rb) as file:data file.read() # print(hashlib.md5(data).hexdigest())# SHA也是摘要算法…...

C高级day四shell脚本
1.思维导图 2.终端输入一个C源文件名(.c结尾)判断文件是否有内容,如果没有内容删除文件,如果有内容编译并执行该文件。 #!/bin/bashread -p "请输入一个.c脚本名:" n if [ -s "$n" ] thenecho $n…...

android studio 写一个小计时器(版本二)
as版本:23.3.1patch2 例程:timer 在前一个版本的基本上改的,增加了继续的功能,实现方法稍微不同。 动画演示: activity_main.xml <?xml version"1.0" encoding"utf-8"?> <androidx…...

【网络安全实验室】SQL注入实战详情
如果额头终将刻上皱纹,你只能做到,不让皱纹刻在你的心上 1.最简单的SQL注入 查看源代码,登录名为admin 最简单的SQL注入,登录名写入一个常规的注入语句: 密码随便填,验证码填正确的,点击登录…...

华为,新华三,思科网络设备指令
1. 设备信息查看 华为 display version # 查看设备版本信息 display device # 查看设备硬件信息 新华三(H3C) display version # 查看设备版本信息 display device # 查看设备硬件信息 锐捷 show version …...

WebRTC线程的启动与运行
WebRTC线程运行的基本逻辑: while(true) {…Get(&msg, …);…Dispatch(&msg);… }Dispatch(Message *pmsg) {…pmsg->handler->OnMessage(pmsg);… }在执行函数内部,就是一个while死循环,只做两件事,从队列里Get取…...

Day3 微服务 微服务保护(请求限流、线程隔离、服务熔断)、Sentinel微服务保护框架、分布式事务(XA模式、AT模式)、Seata分布式事务框架
目录 1.微服务保护 1.1.服务保护方案 1.1.1 请求限流 1.1.2 线程隔离 1.1.3 服务熔断 1.2 Sentinel 1.2.1.介绍和安装 1.2.2 微服务整合 1.2.2.1 引入sentinel依赖 1.2.2.2 配置控制台 1.2.2.3 访问cart-service的任意端点 1.3 请求限流 1.4 线程隔离 1.4.1 OpenFeign整合Senti…...

第9章 子程序与函数调用
汇编语言是一种低级编程语言,它几乎是一对一地映射到计算机的机器码指令。在汇编语言中实现循环结构通常涉及到使用条件跳转指令(如 JMP、JE、JNE 等)来控制程序流程。下面我将通过一个简单的例子来讲解如何用x86汇编语言实现一个循环结构。 …...

manacher算法
Manacher 算法快速入门 Manacher 算法是一种用于寻找字符串中最长回文子串的高效算法,时间复杂度为 O(n)。 基本概念 回文 回文是一个字符串,从左到右和从右到左读都一样。 示例: 回文:"aba"、"abba"非回…...

Cocos2dx Lua绑定生成中间文件时参数类型与源码类型不匹配
这两天维护的一个项目,使用arm64-v8a指令集编译时遇到了报错,提示类型不匹配,具体报错的代码【脚本根据C源文件生成的中间文件】如下: const google::protobuf::RepeatedField<unsigned long long>& ret cobj->equi…...

为什么需要 std::call_once?
std::call_once 是 C 标准库中的一个函数,用来确保某个操作仅被执行一次,通常用于线程安全的初始化操作。它常与 std::once_flag 结合使用,后者用于标记某个操作是否已经执行过。 为什么需要 std::call_once? 在多线程程序中&am…...

ubuntu非root用户操作root权限问题-virbox挂在共享文件夹
首先讲一下,virtuallbox 挂在文件夹,操作的时候总是需要root权限,比较费劲。 这一操作其实也正对着我们在Ubuntu上的操作。 前段时间我想在ubuntu正常用户下去操作i2c,也出现了类似的问题。 后来把正常的操作加到组里面也解决了类…...

网络通讯协议
层次协议应用层HTTP, HTTPS, FTP, SMTP, POP3, IMAP, DNS, DHCP, SNMP, Telnet, SSH, SIP, RTP, RTCP, TFTP, NTP, ICMP, IGMP传输层TCP, UDP网络层IP, ICMP, IGMP数据链路层Ethernet, PPP, HDLC, ATM, Frame Relay ISO/OSI 参考模型协议应用层HTTP, HTTPS, FTP, SMTP, POP3, …...

centos,789使用mamba快速安装devtools
如何进入R语言运行环境请参考:Centos7_miniconda_devtools安装_R语言入门之R包的安装_r语言devtools包怎么安装-CSDN博客 在R里面使用安装devtools经常遇到依赖问题,排除过程过于费时,使用conda安装包等待时间长等。下面演示centos,789都是一…...

【人工智能机器学习基础篇】——深入详解强化学习之常用算法Q-Learning与策略梯度,掌握智能体与环境的交互机制
深入详解强化学习之常用算法:Q-Learning与策略梯度 强化学习(Reinforcement Learning, RL)作为机器学习的一个重要分支,近年来在多个领域取得了显著成果。从棋类游戏的人机对战到自主驾驶汽车,强化学习技术展示了其强大…...

银河麒麟桌面v10sp1修复引导笔记
1.安装双系统最好备份esp分区,uefi引导丢失可以用diskgen,选择工具再点击设置uefi bios,鼠标右键选择efi文件。 2.银河麒麟界面添加windows,复制EFI/Microsoft或者pe生成引导文件后,修复Windows引导用下面命令 /桌面# update-gru…...

深入理解 MVCC 与 BufferPool 缓存机制
深入理解 MVCC 与 BufferPool 缓存机制 在 MySQL 数据库中,MVCC(Multi-Version Concurrency Control)多版本并发控制机制和 BufferPool 缓存机制是非常重要的概念,它们对于保证数据的一致性、并发性以及提升数据库性能起着关键作用…...

vue实现下拉多选、可搜索、全选功能
最后的效果就是树形的下拉多选,可选择任意一级选项,下拉框中有一个按钮可以实现全选,也支持搜索功能。 在mounted生命周期里面获取全部部门的数据,handleTree是讲接口返回的数据整理成树形结构,可以自行解决 <div c…...

探秘Kafka源码:关键内容解析
文章目录 一、以kafka-3.0.0为例1.1安装 gradle 二、生产者源码2.1源码主流程图2.2 初始化2.3生产者sender线程初始化2.4 程序入口2.5生产者 main 线程初始化2.6 跳转到 KafkaProducer构造方法 一、以kafka-3.0.0为例 打开 IDEA,点击 File->Open…->源码包解…...

Android音频效果处理:基于`android.media.audiofx`包的原理、架构与实现
Android音频效果处理:基于android.media.audiofx包的原理、架构与实现 目录 引言Android音频框架概述android.media.audiofx包简介音频效果处理的原理 4.1 音频信号处理基础4.2 常见音频效果android.media.audiofx的架构设计 5.1 类结构分析5.2 设计模式应用系统定制与扩展 6…...

LeetCode - 初级算法 数组(两个数组的交集 II)
两个数组的交集 II 这篇文章讨论如何求两个数组的交集,并返回结果中每个元素出现的次数与其在两个数组中都出现的次数一致。提供多个实现方法以满足不同场景需求。 免责声明:本文来源于个人知识与公开资料,仅用于学术交流。 描述 给定两个整数数组 nums1 和 nums2,以数…...

SQL 实战:分页查询的多种方式对比与优化
在处理大数据表时,分页查询是非常常见的需求。分页不仅可以提高用户体验,还能有效减少数据库查询返回的数据量,避免一次性加载大量记录引起的性能瓶颈。 然而,在数据量较大或复杂查询中,简单的分页方式可能导致性能下降…...
汇川Easy系列正弦信号发生器(ST源代码)
正弦余弦信号发生器CODESYS和MATLAB实现请参考下面文章链接: 正弦余弦信号发生器应用(CODESYS ST源代码+MATLAB仿真)_st语言根据输入值,形成正弦点-CSDN博客文章浏览阅读410次。本文介绍了如何在CODESYS编程环境中创建正弦和余弦信号发生器。通过详细的PLC梯形图和SCL语言代码…...

JavaSpring AI与阿里云通义大模型的集成使用Java Data Science Library(JDSL)进行数据处理
前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默, 忍不住分享一下给大家。点击跳转到网站 学习总结 1、掌握 JAVA入门到进阶知识(持续写作中……) 2、学会Oracle数据库入门到入土用法(创作中……) 3、手把…...

Three.js教程002:Three.js结合Vue进行开发
文章目录 Three.js结合Vue开发创建Vue项目安装依赖运行项目安装three使用three.js完整代码下载Three.js结合Vue开发 创建Vue项目 创建命令: npm init vite@latest框架这里选择【Vue】: 安装依赖 安装命令: cd 01-vueapp npm install运行项目 npm run dev...