设计模式 创建型 建造者模式(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)作为机器学习的一个重要分支,近年来在多个领域取得了显著成果。从棋类游戏的人机对战到自主驾驶汽车,强化学习技术展示了其强大…...

eNSP-Cloud(实现本地电脑与eNSP内设备之间通信)
说明: 想象一下,你正在用eNSP搭建一个虚拟的网络世界,里面有虚拟的路由器、交换机、电脑(PC)等等。这些设备都在你的电脑里面“运行”,它们之间可以互相通信,就像一个封闭的小王国。 但是&#…...
生成xcframework
打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...
【网络】每天掌握一个Linux命令 - iftop
在Linux系统中,iftop是网络管理的得力助手,能实时监控网络流量、连接情况等,帮助排查网络异常。接下来从多方面详细介绍它。 目录 【网络】每天掌握一个Linux命令 - iftop工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景…...
在鸿蒙HarmonyOS 5中实现抖音风格的点赞功能
下面我将详细介绍如何使用HarmonyOS SDK在HarmonyOS 5中实现类似抖音的点赞功能,包括动画效果、数据同步和交互优化。 1. 基础点赞功能实现 1.1 创建数据模型 // VideoModel.ets export class VideoModel {id: string "";title: string ""…...

【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...
连锁超市冷库节能解决方案:如何实现超市降本增效
在连锁超市冷库运营中,高能耗、设备损耗快、人工管理低效等问题长期困扰企业。御控冷库节能解决方案通过智能控制化霜、按需化霜、实时监控、故障诊断、自动预警、远程控制开关六大核心技术,实现年省电费15%-60%,且不改动原有装备、安装快捷、…...
使用van-uploader 的UI组件,结合vue2如何实现图片上传组件的封装
以下是基于 vant-ui(适配 Vue2 版本 )实现截图中照片上传预览、删除功能,并封装成可复用组件的完整代码,包含样式和逻辑实现,可直接在 Vue2 项目中使用: 1. 封装的图片上传组件 ImageUploader.vue <te…...

ETLCloud可能遇到的问题有哪些?常见坑位解析
数据集成平台ETLCloud,主要用于支持数据的抽取(Extract)、转换(Transform)和加载(Load)过程。提供了一个简洁直观的界面,以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...

SpringTask-03.入门案例
一.入门案例 启动类: package com.sky;import lombok.extern.slf4j.Slf4j; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cache.annotation.EnableCach…...

回溯算法学习
一、电话号码的字母组合 import java.util.ArrayList; import java.util.List;import javax.management.loading.PrivateClassLoader;public class letterCombinations {private static final String[] KEYPAD {"", //0"", //1"abc", //2"…...