建造者模式Builder——优雅的使用姿势

在面向对象设计中,建造者模式(Builder Pattern) 是一种非常经典的设计模式,特别适用于需要构造复杂对象的场景。Lombok 提供的 @Builder 注解极大简化了 Builder 模式的实现,而 toBuilder = true 则进一步增强了它的灵活性,使我们能够基于已有对象快速创建新的变体。

本文将深入探讨 @Builder 背后的设计模式,适用场景、不适用场景,以及 @Builder(toBuilder = true) 的使用方法。


一、Builder 模式简介

Builder 模式 是一种创建型设计模式,主要用于构造具有多个可选参数或复杂结构的对象。其核心思想是通过一个 Builder 类一步步设置对象的属性,最后构造出一个完整的对象。

传统 Builder 模式的实现

以构建一个复杂的 Car 类为例,传统的 Builder 模式如下:

public class Car {private final String engine;private final int seats;private final String color;private Car(CarBuilder builder) {this.engine = builder.engine;this.seats = builder.seats;this.color = builder.color;}public static class CarBuilder {private String engine;private int seats;private String color;public CarBuilder engine(String engine) {this.engine = engine;return this;}public CarBuilder seats(int seats) {this.seats = seats;return this;}public CarBuilder color(String color) {this.color = color;return this;}public Car build() {return new Car(this);}}
}

使用方式:

Car car = new Car.CarBuilder().engine("V8").seats(4).color("Red").build();

虽然功能强大,但手动编写 Builder 模式的代码往往繁琐冗长。Lombok 的 @Builder 注解可以让我们完全摆脱这一繁琐过程。


二、Lombok 的 @Builder 简化实现

Lombok 的 @Builder 注解会自动为目标类生成一个静态内部类,作为 Builder 类,同时提供链式方法和 build() 方法。

基本用法

import lombok.Builder;
import lombok.ToString;@Builder
@ToString
public class Car {private final String engine;private final int seats;private final String color;
}public class Main {public static void main(String[] args) {Car car = Car.builder().engine("V8").seats(4).color("Red").build();System.out.println(car);}
}

输出:

Car(engine=V8, seats=4, color=Red)

可以看到,@Builder 自动生成了 CarBuilder 类,并支持链式调用,大幅简化了代码。


三、@Builder(toBuilder = true) 的增强功能

@Builder 配置了 toBuilder = true 时,Lombok 会为目标类生成一个 toBuilder() 方法,允许基于已有对象创建一个新的 Builder。这种增强功能特别适用于需要对不可变对象进行部分修改的场景。

示例代码

@Builder(toBuilder = true)
@ToString
public class Car {private final String engine;private final int seats;private final String color;
}public class Main {public static void main(String[] args) {// 创建初始对象Car car = Car.builder().engine("V8").seats(4).color("Red").build();// 基于现有对象修改部分字段Car updatedCar = car.toBuilder().color("Blue").build();System.out.println("Original Car: " + car);System.out.println("Updated Car: " + updatedCar);}
}

输出:

Original Car: Car(engine=V8, seats=4, color=Red)
Updated Car: Car(engine=V8, seats=4, color=Blue)

toBuilder() 方法的作用:

  • 返回一个 Builder 对象,其中的属性初始值为当前对象的值。
  • 可以灵活修改部分属性值,构造新的对象。

四、Builder 模式的适用场景

1. 需要构造复杂对象时

当一个类的构造方法包含多个参数,特别是可选参数时,使用 Builder 模式可以避免构造器参数混乱问题。

例如,构造一个具有多个可选属性的 Car 类:

Car car = Car.builder().engine("V8").seats(4).color("Red").build();

2. 需要不可变对象时

Builder 模式是构造不可变对象的最佳选择。通过 final 修饰属性和 @Builder,可以确保对象一旦创建,所有字段值都不可更改。

3. 需要对象的灵活更新时

配合 toBuilder = true,可以在保持不可变特性的同时灵活更新对象。例如:

Car updatedCar = car.toBuilder().seats(5).build();

这种场景常见于配置类、请求对象、数据模型等需要频繁修改但又需要保持线程安全的地方。


五、Builder 模式的不适用场景

尽管 Builder 模式功能强大,但它并不适合所有情况:

1. 简单对象的构造

对于只有少量字段的类,使用 Builder 反而显得繁琐。例如:

class User {private String name;private int age;public User(String name, int age) {this.name = name;this.age = age;}
}

对于这样的简单对象,直接使用构造方法更简洁高效。

2. 对象生命周期短,频繁创建

在高性能场景中,如果对象生命周期非常短,且需要频繁创建,Builder 的链式调用可能带来额外开销。

3. 过度设计的风险

如果类的复杂性并不需要 Builder 模式,但仍然引入 Builder,则可能导致代码的复杂性增加。例如,对某些工具类、纯数据类使用 Builder 可能是过度设计。


六、总结

1. @Builder 与设计模式

  • Builder 模式非常适合构造复杂对象或不可变对象。
  • Lombok 的 @Builder 大幅简化了 Builder 模式的实现,让开发者专注于业务逻辑。

2. toBuilder = true 的增强

  • 提供了灵活性,允许基于现有对象创建新的变体。
  • 在不可变对象的更新场景中尤为实用。

3. 适用与不适用场景

  • 适用场景:复杂对象构造、不可变对象、多参数类。
  • 不适用场景:简单类、高性能频繁创建对象场景、过度设计。

4. 推荐实践

  • 对于复杂业务实体或配置类,使用 @BuildertoBuilder = true 是最佳实践。
  • 对于简单类或数据模型,直接使用构造方法或工厂方法即可。

通过合理地选择和应用 Builder 模式,可以有效提升代码的可读性、可维护性以及灵活性。善用 Lombok 的 @Builder,可以让你的代码更加优雅高效!

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

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

相关文章

jenkins 2.346.1最后一个支持java8的版本搭建

1.jenkins下载 下载地址:Index of /war-stable/2.346.1 2.部署 创建目标文件夹,移动到指定位置 创建一个启动脚本,deploy.sh #!/bin/bash set -eDATE$(date %Y%m%d%H%M) # 基础路径 BASE_PATH/opt/projects/jenkins # 服务名称。同时约定部…

Windows10+VirtualBox+Ubuntu:安装虚拟机VirtualBox,虚拟机中安装Ubuntu

一、需求 在Windows10系统中,安装虚拟机VirtualBox,VirtualBox中安装Ubuntu桌面版。 二、环境准备 系统环境 Windows10 内存:8G 虚拟化 虚拟机的运行,如果需要Windows系统开启虚拟化,可以通过BIOS设置。 “虚拟…

pcb元器件选型与焊接测试时的一些个人经验

元件选型 在嘉立创生成bom表,对照bom表买 1、买电容时有50V或者100V是它的耐压值,注意耐压值 2、在买1117等降压芯片时注意它降压后的固定输出,有那种可调降压比如如下,别买错了 贴片元件焊接 我建议先薄薄的在引脚上涂上锡膏…

【漏洞复现】|百易云资产管理运营系统/mobilefront/c/2.php前台文件上传

漏洞描述 湖南众合百易信息技术有限公司(简称:百易云)成立于2017年是一家专注于不动产领域数字化研发及服务的国家高新技术企业,公司拥有不动产领域的数字化全面解决方案、覆盖住宅、写字楼、商业中心、专业市场、产业园区、公建、…

重学 Android 自定义 View 系列(八):星星评分控件(RatingBar)

前言 本节实现一个常见的星星评分控件,广泛应用于各种评价类应用中,比如电影评分、商品评价等。难度不大,直接开搂! 最终效果如下: 1. 效果分析 显示若干颗星星(默认为5颗,可根据属性配置&a…

【力扣热题100】—— Day3.相交链表

被你改变的那部分我,代替你,永远与我站在一起 —— 24.11.28 160. 相交链表 给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回 null 。 图示两个链表在节点 c1 …

SpringBoot实战(三十二)集成 ofdrw,实现 PDF 和 OFD 的转换、SM2 签署OFD

目录 一、OFD 简介1.1 什么是 OFD?1.2 什么是 版式文档?1.3 为什么要用 OFD 而不是PDF? 二、ofdrw 简介2.1 定义2.2 Maven 依赖2.3 ofdrw 的 13 个模块 三、PDF/文本/图片 转 OFD(ofdrw-conterver)3.1 介绍&#xff1a…

cesium 3Dtiles变量

原本有一个变亮的属性luminanceAtZenith,但是新版本的cesium没有这个属性了。于是 let lightColor 3.0result._customShader new this.ffCesium.Cesium.CustomShader({fragmentShaderText:void fragmentMain(FragmentInput fsInput, inout czm_modelMaterial mate…

Java 语言的起源发展与基本概念(JDK,JRE,JVM)

Java语言的起源 源起 Java语言最初是由Sun Microsystems公司(该公司于2009年被Oracle公司收购)开发的一种编程语言。其创造者是詹姆斯高斯林(James Gosling),他是一位加拿大计算机科学家。其前身名为Oak(橡…

Mac安装及合规无限使用Beyond Compare

文章目录 Beyond CompareBeyond Compare简介Beyond Compare安装Beyond Compare到期后继续免费使用 Beyond Compare Beyond Compare简介 Beyond Compare 是一款由 Scooter Software 开发的文件和文件夹比较工具。它主要用于对比两个文件或文件夹之间的差异,并支持文…

使用 Spring AI + Elasticsearch 让 RAG 变得简单

作者:来自 Elastic Laura Trotta 使用私人数据定制你的人工智能聊天机器人体验。 Spring AI 最近将 Elasticsearch 添加为向量存储,Elastic 团队为其提供了优化。我们很高兴展示使用 Spring AI 和 Elasticsearch 向量数据库(vector database&…

C语言:深入理解指针

一.内存和地址 我们知道计算机上CPU(中央处理器)在处理数据的时候,需要的数据是在内存中读取的,处理后的数据也会放回内存中,那我们买电脑的时候,电脑上内存是 8GB/16GB/32GB 等,那这些内存空间…

Spring Boot整合EasyExcel

文章目录 EasyExcel简介Spring Boot整合EasyExcel一、单sheet写操作二、多sheet写数据三、读操作 EasyExcel简介 1、EasyExcel 是一个基于 Java 的简单、省内存的读写 Excel 的开源项目。在尽可能节约内存的情况下支持读写百 M 的 Excel(没有一次性将数据读取到内存…

Windsurf可以上传图片开发UI了

背景 曾经羡慕Cursor的“画图”开发功能,这不Windsurf安排上了。 Upload Images to Cascade Cascade now supports uploading images on premium models Ask Cascade to build or tweak UI from on image upload New keybindings Keybindings to navigate betwe…

Linux中使用ping提示“未知的名称或服务”

Linux中使用ping提示“未知的名称或服务” 问题:在linux系统中使用ping、telnet命令提示“未知的名称或服务”或 bad address。以centos系统为例: 问题原因: 1、未安装ping服务 2、操作系统未设置DNS(尝试ping IP地址&#xff0…

【C++】深入解析 using namespace std 语句

博客主页: [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 💯前言💯什么是 std?💯using namespace std; 的作用💯为什么需要 std 命名空间?💯using namespace std; 的优缺点优点缺点…

Android音频框架总结

1、AudioFlinger:接收多个APP的数据,合并下发;是策略的执行者,例如具体如何与音频设备通信,如何维护现有系统中的音频设备,以及多个音频流的混音如何处理等等都得由它来完 成。 AudioFlinger主要包含3个主…

Jenkins Nginx Vue项目自动化部署

目录 一、环境准备 1.1 Jenkins搭建 1.2 NVM和Nodejs安装 1.3 Nginx安装 二、Jenkins配置 2.1 相关插件安装 2.2 全局工具安装 2.3 环境变量配置 2.4 邮箱配置(构建后发送邮件) 2.5 任务配置 三、Nginx配置 3.1 配置路由转发 四、部署项目 …

BASLER工业相机维修不能触发拍照如何处理解决这个问题

BASLER工业相机维修不能触发拍照如何处理解决这个问题?最近遇到挺多工业相机维修咨询这个不能触发拍照的案例,所以今天优米佳维修的技术就抽空整理了这篇关于BASLER相机不能触发拍照的处理方法分享给大家。 当碰到巴斯勒工业相机不能触发拍照的问题&…

68000汇编实战01-编程基础

文章目录 简介产生背景应用领域 语言学习EASy68K帮助文档IDE使用 编程语言commentslabels开始标签指令标签位置标签 opcode 操作码常用操作码数据传送算术运算逻辑运算控制流分支跳转地址跳转子程序跳转 位操作比较堆栈操作 IO操作码其他操作码 directives 指令DC指令EQU 指令S…