Effective Java 2 遇到多个构造器参数时要考虑使用构建器

news2024/10/6 16:25:49

第2个经验法则:用遇到多个构造器参数时要考虑使用构建器(consider a builder when faced with many constructor parameters)

上一条讨论了静态工厂相对于构造器来说有五大优势。但静态工厂和构造器有个共同的局限性:它 们都不能很好地扩展到大量的可选参数。

对于需要多参数的类,应该用哪种构造器或者静态工厂来编写呢? 接下来,我将通过Java代码示例来对比分析构造器模式、JavaBeans模式以及建造者模式在处理多参数情况下的应用。

构造器模式

假设我们要创建一个 Car 类,它有颜色、品牌、型号和价格等属性,其中颜色和品牌是必需的, 而型号和价格是可选的。

这其实就是重叠构造器(telescoping constructor)模式。程序员一向习惯采用这种模式,在这种模式下,提供的第一个构造器只有必要的参数,第二个构造器有一个可选参数,第三个构造器有两个可选参数,依此类推,最后一个构造器包含所有可选的参数。随着参数增多,构造器的重载会变得复杂且难以管理。

简而言之,重叠构造器模式可行,但是当有许多参数的时候,客户端代码会很难编写并且仍然较 难以阅读。如果读者想知道那些值是什么意思,必须很仔细地数着这些参数来探个究竟。一长串 类型相同的参数会导致一些微妙的错误。如果客户端不小心颠倒了其中两个参数的顺序,编译器 也不会出错,但是程序在运行时会出现错误的行为。

JavaBeans模式

采用JavaBeans模式,我们首先定义一个无参构造器,然后通过setter方法设置属性。

这种模式弥补了重看构造器模式的不足。说得明白一点,就是创建实例很容易,这样产生的代码 读起来也很容易。

遗憾的是,JavaBeans模式自身有着很严重的缺点。因为构造过程被分到了几个调用中在构造过 程中,JavaBean 可能处于不一致的状态,导致对象在完全配置前处于不一致状态。尤其是在多线程环境下或构建过程较长的情境中。下面通过一个具体例子来进一步说明这一问题:

假设我们有一个 Order 类,用于表示在线商店中的订单信息,包括客户ID、商品列表、总价等属 性。使用JavaBeans模式,类定义如下:

假设在某个服务中,我们打算创建一个订单并填充相关信息,但这个过程是分步进行的,可能涉 及多个操作或方法调用,如下所示:

在这个例子中,如果在设置完商品ID之后,程序因为某些原因(如异常抛出、线程切换)没有机 会执行设置总价的逻辑,那么 Order对象就被留在了一个不一致的状态:它有客户ID和商品列 表,但缺少了总价信息。如果此时对象被其他部分的系统使用,可能会引发逻辑错误或计算问题。

另外,在多线程环境下,如果不加锁或其他同步措施,多个线程同时调用setter方法设置不同属性,还可能引起竞态条件,进一步加剧数据的不一致性。

因此,虽然JavaBeans模式提供了灵活性,但在构建过程中必须谨慎管理对象状态的完整性,特 别是在多步骤或多线程场景中,以避免数据不一致的问题。相比之下,建造者模式在这种情况下 提供了更好的解决方案,因为它能确保对象在构建完成之前是一个完整且一致的状态。

建造者模式 (Builder Pattern)

建造者模式通过引入Builder类来解决上述问题(既能保证像重看构造器模式那样的安全性,也能保证像JavaBeans模式那么好的可读性),保持了代码的清晰度和对象的完整性。

建造者模式通过链式调用来设置参数,既保证了代码的可读性,也确保了对象的完整性,尤其适 合参数较多且有可选参数的情况。

它不直接生成想要的对象,而是让客户端利用所有必要的参数调用构造器(或者静态工厂),得到 一个builder对象。然后客户端在 builder 对象上调用类似于 setter 的方法,来设置每个相关的可选参数。最后,客户端调用无参的build方法来生成通常是不可变的对象。

在这个过程中,每次创建Car对象,都会先实例化一个Car.CarBuilder对象,然后通过一系列链 式调用来设置属性,最后调用build()方法生成Car对象。虽然Builder模式提供了清晰的构建 逻辑和良好的可读性,但每个Car对象的创建实际上涉及了两次对象实例化:一次是Builder对 象,一次是最终的Car对象。

想象一个高性能的金融交易系统,每秒需要处理数百万次交易请求。为了优化内存使用和减少GC(垃圾回收)的压力,系统中的每一个环节都需尽可能地高效。在这样的系统中,交易对象的创建频繁且量大,哪怕是最小的性能损失也可能在大规模操作中被放大。

Builder模式不仅可以应用于单个类的复杂对象创建,也非常适合应用于类层次结构,以保持代码的一致性和扩展性。下面通过一个电子产品类层次结构的例子来说明这一点:假设我们有一个基本的Electronics类,以及它的两个子类Smartphone和Laptop,每个类都有其特有的属性。

首先,我们定义一个基础的Electronics类和对应的ElectronicsBuilder。抽象的Electronics类代表了一般的电子产品,包含了一些基础属性,比 如品牌(brand)、型号(model)和价格(price)。同时定义了一个内部抽象类 ElectronicsBuilder,作为构建电子产品实例的模板。这个Builder类定义了一些通用的设置方法(如设置品牌、型号和价格),并且通过泛型参数 T 来确保Builder自身类型的安全返回,即所谓的 fluent interface(流畅接口)设计,让设置过程可以链式调用。

接下来,定义Smartphone类,它继承自Electronics,并新增了特有的属性 storageCapacity。相应的,我们也会创建一个SmartphoneBuilder来构建Smartphone实例。 SmartphoneBuilder继承自E lectronicsBuilder 。 SmartphoneBuilder 除了继承来的通用设置方法外,还添加了一个设置 存储容量的方法。通过覆写 self() 方法返回当前Builder的类型,确保了类型安全和链式调用的延续。

同样地,定义Laptop类,对应的LaptopBuilder负责构建Laptop实例。Laptop类有自己的特性属性——屏幕尺寸 (screenSize )。对应的 LaptopBuilder 同样继承自ElectronicsBuilder的方法,并实现了自己的 self() 和 ElectronicsBuilder ,添加了设置屏幕尺 build() 方法,以适应 Laptop 的构建需求。

现在,可以这样使用Builder模式来创建不同类型的电子产品,并保持代码的清晰和类型安全:

通过这种方式,Builder模式不仅解决了复杂对象的构建问题,而且在类层次结构中保持了良好的扩展性和一致性,使得每个子类都能拥有自己特性的Builder,同时复用了基类Builder的部分逻辑,减少了代码重复,提升了代码的可维护性。

Builder 模式的确也有它自身的不足:

为了创建对象,必须先创建它的构建器。虽然创建这个构建器的开销在实践中可能不那么明显,但是在某些十分注重性能的情况下,可能就成问题了。 

Builder模式还比重看构造器模式更加冗长,因此它只在有很多参数的时候才使用,比如4个 或者更多个参数。

简而言之,如果类的构造器或者静态工厂中具有多个参数,设计这种类时,Builder模式就是一 种不错的选择。 特别是当大多数参数都是可选或者类型相同的时候。与使用重叠构造器模式相比,使用Builder 模式的客户端代码将更易干阅读和编写,构建器也比JavaBeans加安全。

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

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

相关文章

SpringBoot——整合WebSocket长连接

目录 WebSocket 项目总结 新建一个SpringBoot项目 pom.xml WebSocketConfig配置类 TestWebSocketEndpoint服务端点类 socket.html客户端 IndexController控制器 SpringbootWebsocketApplication启动类 测试客户端和服务端如何使用WebSocket进行连接和通信 WebSocket S…

LayUI的暗淡:错误的押宝了前后端不分离

LayUI是一个不错的中后台UI框架,贝格前端工场用的CMS就是基于layUI的,可以说简单轻便。除此之外,贝格前端工场很少接到客户要求升级LayUI界面,或者采用LayUI框架的。 一、LayUI官网的谢幕,吹起了前后端不分离模式没落…

Java Set系列集合的使用规则和场景(HashSet,LinkedHashSet,TreeSet)

Set集合 package SetDemo;import java.util.HashSet; import java.util.Iterator; import java.util.Set;public class SetDemo {public static void main(String[] args) {/*Set集合的特点:1.Set系列集合的特点:Set集合是一个存储元素不能重复的集合方…

【数据结构(邓俊辉)学习笔记】图03——拓扑排序

文章目录 0. 概述1. 零入度算法1. 1 拓扑排序1. 2 算法 2. 零出度算法2.1 算法2.2 实现2.3. 复杂度 0. 概述 学习下拓扑排序 1. 零入度算法 1. 1 拓扑排序 首先理解下拓扑排序 其实老师经常干这事,如编讲义,将已经知道的知识点串起来变成讲课序列。那…

AIGC作答《2024年高考作文|新课标I卷》能拿多少分?

AIGC作答《2024年高考作文|新课标I卷》能拿多少分? 一、前言二、题目三、作答 一、前言 如火如荼的2024年高考圆满落幕,在如此Happy的时刻,AIGC技术正以其前所未有的热度席卷全球。它不仅改变了我们获取信息的方式,也…

RJ45 PCB布线

RJ45底盘接地和数字地通过一个1M欧姆的电阻和一个0.1uF的去耦电容隔离。其底盘接地和数字地的间距,必须比60mil宽。如图11及图12所示。 图11 典型变压器集成单RJ45的机箱/数字地平面 图12 典型RJ45和变压器分开的机箱/数字地平面https://www.bilibili.com/read/…

Java学习-JDBC(二)

核心API 注册驱动 Class.forName(“com.mysql.cj.jdbc.Driver”);在Java中,当使用JDBC连接数据库时,需要加载数据库特定的驱动程序,以便与数据库进行通信,加载驱动程序的目的是为了注册驱动程序,使得JDBC API能够识别…

Element-UI入门

目录 1.什么是Element-UI 2.作用 3.版本历史 4.优缺点 4.1.优点 4.2.缺点 5.应用场景 6.代码示例 7.未来展望 8.总结 1.什么是Element-UI Element-UI 是由饿了么前端团队开发的一套基于 Vue.js 的桌面端组件库。提供了一整套 UI 组件,使开发者能够快速构…

统信UOS1070上配置文件管理器默认属性01

原文链接:统信UOS 1070上配置文件管理器默认属性01 Hello,大家好啊!今天给大家带来一篇关于在统信UOS 1070上配置文件管理器默认属性的文章。文件管理器是我们日常操作系统使用中非常重要的工具,了解如何配置其默认属性可以极大地…

使用贝塞尔曲线实现一个iOS时间轴

UI效果 实现的思路 就是通过贝塞尔曲线画出时间轴的圆环的路径,然后 使用CAShaper来渲染UI,再通过 animation.beginTime [cilrclLayer convertTime:CACurrentMediaTime() fromLayer:nil] circleTimeOffset 来设置每个圆环的动画开始时间, …

手写kNN算法的实现-用余弦相似度来度量距离

设a为预测点,b为其中一个样本点,在向量空间里,它们的形成的夹角为θ,那么θ越小(cosθ的值越接近1),就说明a点越接近b点。所以我们可以通过考察余弦相似度来预测a点的类型。 from collections i…

【学术小白成长之路】03三方演化博弈(基于复制动态方程)均衡点与稳定性分析

从本专栏开始,笔者正式研究演化博弈分析,其中涉及到双方演化博弈分析,三方演化博弈分析,复杂网络博弈分析等等。 先阅读了大量相关的博弈分析的文献,总结了现有的研究常用的研究流程,针对每个流程进行拆解。…

HTML-CSS练习例子

HTML CSS 练习 https://icodethis.com 作为前端练习生。不敲代码只看,入门是很慢的,所以直接实战是学习前端最快的途径之一。 这个网站练习HTML CSS的,可以打开了解一下,可以每天打卡,例子简单,循序渐进&…

《TCP/IP网络编程》(第十三章)多种I/O函数(2)

使用readv和writev函数可以提高数据通信的效率,它们的功能可以概括为**“对数据进行整合传输及发送”**。 即使用writev函数可以将分散在多个缓冲中的数据一并发送,使用readv函数可以由多个缓冲分别接受,所以适当使用他们可以减少I/O函数的调…

Pytorch 实现目标检测一(Pytorch 23)

一 目标检测和边界框 在图像分类任务中,我们假设图像中只有一个主要物体对象,我们只关注如何识别其类别。然而,很多时候图像里有多个我们感兴趣的目标,我们不仅想知 道它们的类别,还想得到它们在图像中的具体位置。在…

ESP8266+STM32+阿里云保姆级教程(AT指令+MQTT)

前言:在开发过程中,几乎踩便了所有大坑小坑总结出的文章,我是把坑踩满了,帮助更过小白快速上手,如有错误之处,还麻烦各位大佬帮忙指正、 目录 一、ESP-01s介绍 1、ESP-01s管脚功能: 模组启动模…

vscode 突然无法启动 WSL terminal 了怎么办?

参考:https://github.com/microsoft/vscode/issues/107485 根据参考网页,似乎在 windows 更新之后,重启,就有可能出现标题所说的 vscode 无法启动 WSL terminal 的情况。 首先使用 cmd 进入 wsl 终端,把 ~/.vscode-se…

EON安装ASE Interface

EON安装 我的eon路径于/eon/。 则环境为 export PYTHONPATH/eon/:$PYTHONPATH export PATH/eon/bin:$PATHsource: https://theory.cm.utexas.edu/eon/installation.html ASE 测试系统ubuntu。如果你python2和python3总是纠缠不清,可以sudo apt install python-…

vscode中执行python语句dir(torch)不返回结果

输入半天,发现在IDLE运行后的shell界面输入语句就会返回一大串。但是在vscode中老是不返回值。 结果恍然发现这没加print()。 无语惨了。 家人们,这是python,而不是matlab。思维还没转换过来,笑死

扩散模型Stable Diffusion

扩散模型构成 Text Encoder(CLIPText) Clip Text为文本编码器。以77 token为输入,输出为77 token 嵌入向量,每个向量有768维度。 Diffusion(UNetScheduler) 在潜在空间中逐步处理扩散信息。以文本嵌入向量和由噪声组成的起始多维数组为输入&#xff0c…