Java编程之建造者模式

建造者模式(Builder Pattern)是一种创建型设计模式,它将一个复杂对象的构建与表示分离,使得同样的构建过程可以创建不同的表示。这种模式允许你分步骤构建一个复杂对象,并且可以在构建过程中进行不同的配置。

模式的核心组件

建造者模式通常包含以下四个核心组件:

  1. 产品(Product):要构建的复杂对象。
  2. 抽象建造者(Builder):定义了构建产品各个部分的抽象接口。
  3. 具体建造者(Concrete Builder):实现抽象建造者接口,完成产品各部分的具体构建。
  4. 指挥者(Director):负责安排复杂对象的构建顺序,并使用建造者对象构建产品。

简单实现

让我们通过一个示例来详细说明建造者模式的实现。假设我们要构建一个电脑(Computer)对象,它包含 CPU、内存、硬盘和显卡等组件。

首先,定义产品类:

// 产品类:电脑
public class Computer {private String cpu;private String memory;private String hardDisk;private String graphicsCard;// 私有构造函数,只能通过Builder创建Computerprivate Computer(Builder builder) {this.cpu = builder.cpu;this.memory = builder.memory;this.hardDisk = builder.hardDisk;this.graphicsCard = builder.graphicsCard;}// Getter方法public String getCpu() {return cpu;}public String getMemory() {return memory;}public String getHardDisk() {return hardDisk;}public String getGraphicsCard() {return graphicsCard;}@Overridepublic String toString() {return "Computer{" +"cpu='" + cpu + '\'' +", memory='" + memory + '\'' +", hardDisk='" + hardDisk + '\'' +", graphicsCard='" + graphicsCard + '\'' +'}';}// 静态内部类:Builderpublic static class Builder {private String cpu;private String memory;private String hardDisk;private String graphicsCard;// 设置CPU,返回Builder实例以便链式调用public Builder setCpu(String cpu) {this.cpu = cpu;return this;}// 设置内存,返回Builder实例以便链式调用public Builder setMemory(String memory) {this.memory = memory;return this;}// 设置硬盘,返回Builder实例以便链式调用public Builder setHardDisk(String hardDisk) {this.hardDisk = hardDisk;return this;}// 设置显卡,返回Builder实例以便链式调用public Builder setGraphicsCard(String graphicsCard) {this.graphicsCard = graphicsCard;return this;}// 构建Computer对象public Computer build() {// 可以在这里添加参数验证逻辑if (cpu == null || memory == null || hardDisk == null) {throw new IllegalArgumentException("CPU、内存和硬盘是必需的");}return new Computer(this);}}
}

然后,使用建造者模式创建电脑对象:

public class BuilderPatternExample {public static void main(String[] args) {// 使用Builder创建电脑对象Computer gamingComputer = new Computer.Builder().setCpu("Intel i9-12900K").setMemory("32GB DDR5").setHardDisk("2TB NVMe SSD").setGraphicsCard("NVIDIA RTX 3080").build();Computer officeComputer = new Computer.Builder().setCpu("Intel i5-12400").setMemory("16GB DDR4").setHardDisk("512GB SSD").build(); // 办公电脑可以不设置独立显卡System.out.println("游戏电脑配置:" + gamingComputer);System.out.println("办公电脑配置:" + officeComputer);}
}

如上是一个建造者模式的简单实现,是不是感觉超级简单,下面是经典的实现方式

 经典实现

// 1. 产品类:电脑
public class Computer {private String cpu;private String memory;private String hardDisk;private String graphicsCard;// 构造函数和Getter方法public Computer(String cpu, String memory, String hardDisk, String graphicsCard) {this.cpu = cpu;this.memory = memory;this.hardDisk = hardDisk;this.graphicsCard = graphicsCard;}@Overridepublic String toString() {return "Computer{" +"cpu='" + cpu + '\'' +", memory='" + memory + '\'' +", hardDisk='" + hardDisk + '\'' +", graphicsCard='" + graphicsCard + '\'' +'}';}
}// 2. 抽象建造者接口
public interface ComputerBuilder {void buildCPU();void buildMemory();void buildHardDisk();void buildGraphicsCard();Computer getComputer();
}// 3. 具体建造者:游戏电脑建造者
public class GamingComputerBuilder implements ComputerBuilder {private Computer computer;public GamingComputerBuilder() {this.computer = new Computer(null, null, null, null);}@Overridepublic void buildCPU() {computer.setCpu("Intel i9-12900K");}@Overridepublic void buildMemory() {computer.setMemory("32GB DDR5");}@Overridepublic void buildHardDisk() {computer.setHardDisk("2TB NVMe SSD");}@Overridepublic void buildGraphicsCard() {computer.setGraphicsCard("NVIDIA RTX 3080");}@Overridepublic Computer getComputer() {return computer;}
}// 3. 具体建造者:办公电脑建造者
public class OfficeComputerBuilder implements ComputerBuilder {private Computer computer;public OfficeComputerBuilder() {this.computer = new Computer(null, null, null, null);}@Overridepublic void buildCPU() {computer.setCpu("Intel i5-12400");}@Overridepublic void buildMemory() {computer.setMemory("16GB DDR4");}@Overridepublic void buildHardDisk() {computer.setHardDisk("512GB SSD");}@Overridepublic void buildGraphicsCard() {// 办公电脑可以不设置独立显卡}@Overridepublic Computer getComputer() {return computer;}
}// 4. 指挥者:控制构建过程
public class ComputerDirector {private ComputerBuilder builder;public ComputerDirector(ComputerBuilder builder) {this.builder = builder;}public void constructComputer() {builder.buildCPU();builder.buildMemory();builder.buildHardDisk();builder.buildGraphicsCard();}public Computer getComputer() {return builder.getComputer();}
}// 客户端代码
public class Client {public static void main(String[] args) {// 构建游戏电脑ComputerBuilder gamingBuilder = new GamingComputerBuilder();ComputerDirector director = new ComputerDirector(gamingBuilder);director.constructComputer();Computer gamingComputer = director.getComputer();// 构建办公电脑ComputerBuilder officeBuilder = new OfficeComputerBuilder();director = new ComputerDirector(officeBuilder);director.constructComputer();Computer officeComputer = director.getComputer();System.out.println("游戏电脑配置:" + gamingComputer);System.out.println("办公电脑配置:" + officeComputer);}
}

模式的优点

  1. 分离构建和表示:使得构建算法可以独立于产品的表示,提高了系统的可扩展性。
  2. 分步构建:可以分步骤构建一个复杂对象,允许在构建过程中进行不同的配置。
  3. 链式调用:通过链式调用方法设置属性,使代码更简洁易读。
  4. 参数验证:可以在build()方法中添加参数验证逻辑,确保对象的完整性。

模式的应用场景

  1. 复杂对象构建:当创建一个对象需要很多步骤,并且这些步骤的顺序可能不同。
  2. 可选参数较多:当一个类的构造函数有很多参数,特别是其中很多是可选参数时。
  3. 构建不同表示:当需要创建同一个产品的不同表示时。

与其他模式的比较

  • 工厂模式:工厂模式关注的是整体对象的创建,而建造者模式关注的是对象的分步构建。
  • 抽象工厂模式:抽象工厂模式返回的是一系列相关产品,而建造者模式返回的是一个完整的产品。

建造者模式在实际开发中非常常用,特别是在创建复杂对象时。它能够使代码更加清晰、灵活,并且易于维护。另外Lombok 的@Builder注解是一个强大的工具,它能自动生成流式 API 风格的 "建造者模式 (Builder Pattern)" 代码,从而避免编写大量样板代码。使用该注解可以让代码更简洁,同时保持建造者模式的优势。

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

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

相关文章

Spring AI之RAG入门

目录 1. 什么是RAG 2. RAG典型应用场景 3. RAG核心流程 3.1. 检索阶段 3.2. 生成阶段 4. 使用Spring AI实现RAG 4.1. 创建项目 4.2. 配置application.yml 4.3. 安装ElasticSearch和Kibana 4.3.1. 安装并启动ElasticSearch 4.3.2. 验证ElasticSearch是否启动成功 …

mysql数据库实现分库分表,读写分离中间件sharding-sphere

一 概述 1.1 sharding-sphere 作用: 定位关系型数据库的中间件,合理在分布式环境下使用关系型数据库操作,目前有三个产品 1.sharding-jdbc,sharding-proxy 1.2 sharding-proxy实现读写分离的api版本 4.x版本 5.x版本 1.3 说明…

运维视角下的广告系统之理解广告索引级联

广告索引中为什么要级联 这里的“级联”一般指的是多层索引结构,也叫级联索引(Cascade Index 或 Multi-level Index)。 在广告系统的索引中,级联设计有重要作用,主要原因如下: 1. 多维特征筛选的需求 广…

2025年5月24日系统架构设计师考试题目回顾

当前仅仅是个人用于记录&#xff0c;还未做详细分析&#xff0c;待更新… 综合知识 设 x,y 满足约束条件&#xff1a;x-1>0, x-y<0, x-y-x<0, 则 y/x 的最大值是()。 A. 3 B. 2 C. 4 D. 1 申请软件著作权登记时应当向中国版本保护中心提交软件的鉴别材料&#xff…

3D-激光SLAM笔记

目录 定位方案 编译tbb ros2humble安装 命令 colcon commond not found 栅格地图生成&#xff1a; evo画轨迹曲线 安装gtsam4.0.2 安装ceres-solver1.14.0 定位方案 1 方案一&#xff1a;改动最多 fasterlio 建图&#xff0c;加闭环优化&#xff0c;参考fast-lio增加关…

贪心算法应用:分数背包问题详解

贪心算法与分数背包问题 贪心算法&#xff08;Greedy Algorithm&#xff09;是算法设计中一种重要的思想&#xff0c;它在许多经典问题中展现出独特的优势。本文将用2万字篇幅&#xff0c;深入剖析贪心算法在分数背包问题中的应用&#xff0c;从基础原理到Java实现细节&#x…

PyTorch——非线性激活(5)

非线性激活函数的作用是让神经网络能够理解更复杂的模式和规律。如果没有非线性激活函数&#xff0c;神经网络就只能进行简单的加法和乘法运算&#xff0c;没法处理复杂的问题。 非线性变化的目的就是给我们的网络当中引入一些非线性特征 Relu 激活函数 Relu处理图像 # 导入必…

iOS 电子书听书功能的实现

在 iOS 应用中实现电子书听书&#xff08;文本转语音&#xff09;功能&#xff0c;可以通过系统提供的 AVFoundation 框架实现。以下是详细实现步骤和代码示例&#xff1a; 核心步骤&#xff1a; 导入框架创建语音合成器配置语音参数实现播放控制处理后台播放添加进度跟踪 完整…

ES中must与filter的区别

在 Elasticsearch 的布尔查询&#xff08;bool query&#xff09;中&#xff0c;must 和 filter 是两个核心子句&#xff0c;它们的核心区别在于 是否影响相关性评分&#xff0c;这直接决定了它们在查询性能、使用场景和结果排序上的差异。以下是详细对比&#xff1a; 一、核心…

vscode实时预览编辑markdown

vscode实时预览编辑markdown 点击vsode界面&#xff0c;实现快捷键如下&#xff1a; 按下快捷键 CtrlShiftV&#xff08;Windows/Linux&#xff09;或 CommandShiftV&#xff08;Mac&#xff09;即可在侧边栏打开 Markdown 预览。 效果如下&#xff1a;

Android第十一次面试flutter篇

Flutter基础​ 在 Flutter 中&#xff0c;​三棵树&#xff08;Widget Tree、Element Tree、RenderObject Tree&#xff09;​​ 是框架的核心设计&#xff0c;它们协同工作以实现高效的 UI 渲染和更新机制。 ​1. Widget Tree&#xff08;Widget 树&#xff09;​​ ​是什么…

多线程编程中的数据竞争与内存可见性问题解析

引言 在多线程编程中&#xff0c;看似简单的代码往往隐藏着复杂的并发问题。今天我们来分析一个经典的生产者-消费者场景&#xff0c;看看在多核CPU环境下可能出现的各种"意外"情况。 问题代码分析 让我们先看看这段看似正常的C#代码&#xff1a; using System; u…

Linux 与 Windows:哪个操作系统适合你?

Linux vs Windows:系统选择的关键考量 在数字化转型浪潮中,操作系统作为底层基础设施的重要性日益凸显。Linux与Windows作为主流选择,其差异不仅体现在技术架构上,更深刻影响着开发效率、运维成本与安全性。本文将从​​7个核心维度​​展开对比分析,并提供典型应用场景建…

佰力博科技与您探讨低温介电温谱测试仪的应用领域

低温介电温谱测试应用领域有如下&#xff1a; 一、电子材料&#xff1a; 低温介电温谱测试仪广泛应用于电子材料的性能测试&#xff0c;如陶瓷材料、半导体材料、压电材料等。通过该设备&#xff0c;可以评估材料在高温或低温环境下的介电性能&#xff0c;为材料的优化和应用提…

Windows 下彻底删除 VsCode

彻底删除 VS Code (Visual Studio Code) 意味着不仅要卸载应用程序本身&#xff0c;还要删除所有相关的配置文件、用户数据、插件和缓存。这可以确保你有一个完全干净的状态&#xff0c;方便你重新安装或只是彻底移除它。 重要提示&#xff1a; 在执行以下操作之前&#xff0c…

STM32与GD32标准外设库深度对比

近年来,随着全球芯片短缺和市场价格波动,工程师们开始寻求对常用MCU的替代方案。在STM32因产能受限而频频涨价的背景下,GD32作为国产替代的重要选项,获得了越来越多的关注。尤其是GD32F103系列,由于其在硬件封装、功能特性乃至软件支持上的“高相似度”,成为STM32F103的热…

使用Redis的四个常见问题及其解决方案

Redis 缓存穿透 定义&#xff1a;redis查询一个不存在的数据&#xff0c;导致每次都查询数据库 解决方案&#xff1a; 如果查询的数据为空&#xff0c;在redis对应的key缓存空数据&#xff0c;并设置短TTL。 因为缓存穿透通常是因为被恶意用不存在的查询参数进行压测攻击&…

Java高级 | 【实验一】Spring Boot安装及测试 最新

隶属文章&#xff1a;Java高级 | &#xff08;二十二&#xff09;Java常用类库-CSDN博客 目录 一、SpringBoot的特点 二、Spring Boot安装及测试 &#xff08;一&#xff09;安装Intellij IDEA &#xff08;二&#xff09;安装MySQL &#xff08;三&#xff09;安装postma…

Oracle RMAN自动恢复测试脚本

说明 此恢复测试脚本&#xff0c;基于rman备份脚本文章使用的fullbak.sh做的备份。 数据库将被恢复到RESTORE_LO参数设置的位置。 在恢复完成后&#xff0c;执行一个测试sql,确认数据库恢复完成&#xff0c;数据库备份是好的。恢复测试数据库的参数&#xff0c;比如SGA大小都…

从Java的JDK源码中学设计模式之装饰器模式

装饰器模式是一种极具弹性的结构型设计模式&#xff0c;它允许我们通过组合的方式动态扩展对象功能而无需修改原有结构。本文将通过JDK源码中的实际应用和通俗易懂的代码示例&#xff0c;带你深入了解这一强大模式的精髓。 装饰器模式核心原理 装饰器模式的核心思想&#xff…