网站首页 > java教程 正文
在上一个问题中,我们创建了MelonRecord Java记录,并通过以下代码实例化它:
MelonRecord melonr = new MelonRecord("Cantaloupe", 2600);
这怎么可能(因为我们在MelonRecord中没有写任何参数化构造函数)?编译器只是按照其内部协议为Java记录创建了一个基于我们在记录声明中提供的组件的默认构造函数(在这种情况下,有两个组件,type和weight)。这个构造函数被称为规范构造函数,它总是与给定的组件对齐。每个记录都有一个规范构造函数,这是创建该记录实例的唯一方式。但是,我们可以重新定义规范构造函数。下面是一个类似于默认构造函数的显式规范构造函数 - 如你所看到的,规范构造函数只是接受所有给定的组件并将相应的实例字段设置为(也由编译器生成为私有final字段):
public MelonRecord(String type, float weight) {
this.type = type;
this.weight = weight;
}
一旦实例被创建,它就不能被改变(它是不可变的)。它将唯一的目的在你的程序中携带这些数据。这个显式的规范构造函数有一个快捷方式,称为紧凑构造函数 - 这是Java记录特有的。由于编译器知道给定组件的列表,它可以从这个紧凑构造函数完成其工作,这个构造函数等同于前一个构造函数:
public MelonRecord {}
请注意,不要将这个紧凑构造函数与不带参数的构造函数混淆。以下代码片段是不等价的:
public MelonRecord {} // 紧凑构造函数
public MelonRecord() {} // 无参数构造函数
当然,没有意义去写一个显式的规范构造函数,只是为了模仿默认构造函数的作用。因此,让我们检查几个重新定义规范构造函数有意义的场景。
处理验证
在这个时候,当我们创建一个MelonRecord时,我们可以将类型传递为null,或者将甜瓜的重量作为负数。这导致了包含非有效数据的损坏记录。可以在显式的规范构造函数中处理记录组件的验证,如下所示:
public record MelonRecord(String type, float weight) {
// 用于验证的显式规范构造函数
public MelonRecord(String type, int weight) {
if (type == null) {
throw new IllegalArgumentException(
"The melon's type cannot be null");
}
if (weight < 1000 || weight > 10000) {
throw new IllegalArgumentException("The melon's weight
must be between 1000 and 10000 grams");
}
this.type = type;
this.weight = weight;
}
}
或者,通过紧凑构造函数,如下所示:
public record MelonRecord(String type, float weight) {
// 用于验证的显式紧凑构造函数
public MelonRecord {
if (type == null) {
throw new IllegalArgumentException(
"The melon's type cannot be null");
}
if (weight < 1000 || weight > 10000) {
throw new IllegalArgumentException("The melon's weight
must be between 1000 and 10000 grams");
}
}
}
验证处理是显式规范/紧凑构造函数的最常见用例。接下来,让我们看看两个较少为人所知的用例。
重新分配组件
通过显式的规范/紧凑构造函数,我们可以重新分配组件。例如,当我们创建一个MelonRecord时,我们提供了它的类型(例如,Cantaloupe)和它的重量(例如,2600克)。但是,如果我们想使用重量单位为千克(2600克=2.6千克),那么我们可以在一个显式的规范构造函数中提供这种转换,如下所示:
// 用于重新分配组件的显式规范构造函数
public MelonRecord(String type, float weight) {
weight = weight/1_000; // 覆盖组件 'weight'
this.type = type;
this.weight = weight;
}
如你所看到的,weight组件是可用的,并在weight字段初始化为新分配的值之前重新分配。最后,weight组件和weight字段具有相同的值(2.6千克)。那么这段代码呢?
public MelonRecord(String type, float weight) {
this.type = type;
this.weight = weight/1_000;
}
嗯,在这种情况下,最后weight字段和weight组件将有不同的值。weight字段是2.6千克,而weight组件是2600克。请注意,这很可能不是你想要做的。让我们检查另一段代码:
public MelonRecord(String type, float weight) {
this.type = type;
this.weight = weight;
weight = weight/1_000;
}
再次,最后weight字段和weight组件将有不同的值。weight字段是2600克,而weight组件是2.6千克。再次提醒,这很可能不是你想要做的。当然,更干净和最简单的方法是依赖紧凑构造函数。这一次,我们不能偷偷进行意外的重新分配:
public record MelonRecord(String type, float weight) {
// 用于重新分配组件的显式紧凑构造函数
public MelonRecord {
weight = weight/1_000; // 覆盖组件 'weight'
}
}
最后,让我们处理第三种情况。
给定组件的防御性副本
我们知道Java记录是不可变的。但这并不意味着它的组件也是不可变的。想想数组、列表、映射、日期等组件。所有这些组件都是可变的。为了恢复完全的不可变性,你将更喜欢在这些组件的副本上工作,而不是修改给定的组件。正如你已经直觉到的,这可以通过显式的规范构造函数完成。例如,让我们考虑以下记录,它获取一个表示一组项目的零售价格的单个组件作为Map:
public record MarketRecord(Map<String, Integer> retails) {}
这个记录不应该修改这个Map,所以它依赖于一个显式的规范构造函数来创建一个防御性副本,该副本将在后续任务中使用,没有任何修改的风险(Map.copyOf()返回给定Map的不可修改副本):
public record MarketRecord(Map<String, Integer> retails) {
public MarketRecord {
retails = Map.copyOf(retails);
}
}
基本上,这只是组件重新分配的一种风味。此外,我们可以通过访问器方法返回防御性副本:
public Map<String, Integer> retails() {
return Map.copyOf(retails);
}
// 或者,Java Bean风格的getter
public Map<String, Integer> getRetails() {
return Map.copyOf(retails);
}
猜你喜欢
- 2024-10-17 Java|剖析类内的五类成员:属性、方法、构造器、代码块、内部类
- 2024-10-17 C++|构造、析构、成员(静态、非静态)函数、友元、全局函数
- 2024-10-17 JavaScript构造函数和原型:构造函数和原型 原创
- 2024-10-17 灵魂拷问:创建 Java 字符串,用""还是构造函数
- 2024-10-17 dart系列之:dart类中的构造函数(dart命名构造函数)
- 2024-10-17 Java面试题#构造函数和Setter哪种DI方式更好
- 2024-10-17 「php」construct() 函数介绍与使用方法详解
- 2024-10-17 为什么不允许使用 Java 静态构造函数?
- 2024-10-17 关于构造函数的Java面试问题(关于构造函数的java面试问题有哪些)
- 2024-10-17 编程语言Java如何创建对象,看完秒懂
你 发表评论:
欢迎- 05-16SpringBoot整合Redis实现常用功能
- 05-16基于Redis实现简单的延时消息队列
- 05-16安装Redis
- 05-16Spring系列之Redis的两种集成方式
- 05-16Django连接Redis集群问题排查思路和总结
- 05-16只需5分钟,完成Redis所有命令操作~
- 05-16熟练使用 Redis 的 5 大数据结构:Java 实战教程
- 05-16Redis 常见业务场景及实例(Java)
- 最近发表
- 标签列表
-
- java反编译工具 (77)
- java反射 (57)
- java接口 (61)
- java随机数 (63)
- java7下载 (59)
- java数据结构 (61)
- java 三目运算符 (65)
- java对象转map (63)
- Java继承 (69)
- java字符串替换 (60)
- 快速排序java (59)
- java并发编程 (58)
- java api文档 (60)
- centos安装java (57)
- java调用webservice接口 (61)
- java深拷贝 (61)
- 工厂模式java (59)
- java代理模式 (59)
- java.lang (57)
- java连接mysql数据库 (67)
- java重载 (68)
- java 循环语句 (66)
- java反序列化 (58)
- java时间函数 (60)
- java是值传递还是引用传递 (62)
本文暂时没有评论,来添加一个吧(●'◡'●)