return false;
} else {
Person other = (Person) obj;
return Objects.equals(name, other.name)
&& Objects.equals(address, other.address);
}
}
@Override
public String toString() {
return “Person [name=” + name + “, address=” + address + “]”;
}
// standard getters
}
为了达到我们的需求,我们这里有两个小问题:
1. 这个类里包含太多样板化的代码了
2. 我们为了写一个Person类包含name和address两个字段,写了太多别的东西从而让这个类太复杂了
第一种情况,我们必须对每个这种class重复相同的繁琐过程,为每个数据段单调地创建一个新字段,创建equals、hashCode和toString方法,并创建一个接受每个字段的构造函数。
虽然现在很多IDE可以自动生成这些代码,但是有个问题是它们没办法去在我们修改字段后主动的更新对应生成的代码。比如我新增一个age字段,构造方法、equals方法之类的我都要删了重写,IDE没法自动更新,而且还很容易遗漏或是忘记更新这些方法。
第二种情况,额外的代码掩盖了我们的类只是一个数据类,它有两个字符串字段:name和address。
更好的方法是显式声明我们的类是一个数据类。
3、基础用法
下面就要请出我们的 record 关键字了。
在jdk14,我们可以用record关键字替换重复的数据类。record是不可变的数据类,只需要字段的类型和名称。equals、hashCode和toString方法以及private、final字段和public的构造函数都是由Java编译器生成的。
下面我们来看看jdk14用record关键字怎么创建这个包含name和address的Person。
public record Person (String name, String address) {}
没错,就是这么简单的实现了。
3.1、生成的构造方法
用record关键字,创建出的构造方法如下:
public Person(String name, String address) {
this.name = name;
this.address = address;
}
record生成的构造方法,和一般的class的构造使用方式没什么区别:
Person person = new Person(“John Doe”, “100 Linda Ln.”);
作者ps:其实这里看生成的代码会发现,其实他是创建了个final的Person类,然后继承了java.lang.Record,不过Record类也是final的,我们自己写代码的时候是没法继承的。
3.2、生成的getters方法
用record关键字后,同样为我们生成了getters方法,只不过和我们传统使用的调用有点区别,看下面的测试用例:
@Test
public void givenValidNameAndAddress_whenGetNameAndAddress_thenExpectedValuesReturned() {
String name = “John Doe”;
String address = “100 Linda Ln.”;
Person person = new Person(name, address);
assertEquals(name, person.name());
assertEquals(address, person.address());
}
它生成的是不含get前缀的。
3.3、生成的equals方法
用record关键字后,同样为我们生成的对应的equals方法,测试用例如下:
@Test
public void givenSameNameAndAddress_whenEquals_thenPersonsEqual() {
String name = “John Doe”;
String address = “100 Linda Ln.”;
Person person1 = new Person(name, address);
Person person2 = new Person(name, address);
assertTrue(person1.equals(person2));
}
只要是同一个对象,或是每个字段都相等,就会是true。
3.4、生成的hashCode方法
和equals一样,为我们生成的hashCode方法,测试用例如下:
@Test
public void g
必看视频!获取2024年最新Java开发全套学习资料 备注Java
ivenSameNameAndAddress_whenHashCode_thenPersonsEqual() {
String name = “John Doe”;
String address = “100 Linda Ln.”;
Person person1 = new Person(name, address);
Person person2 = new Person(name, address);
assertEquals(person1.hashCode(), person2.hashCode());
}
当然也是任何的字段不一样就会返回不一样的code值,所有字段都一样返回相同的code值。但是返回相同code值不代表所有字段都一样(毕竟有碰撞概率)。
3.5、生成的toString方法
一个name为“John Doe”,address为“100 Linda Ln.”的Person,toString样例如下:
Person[name=John Doe, address=100 Linda Ln.]
4、构造方法
虽然为我们生成了一个public的构造方法,但我们仍然可以自定义其他构造方法的实现。
此定制旨在用于验证,并应尽可能简单。
比如我们要确保传入的name和address两个字段都不为空,那么我们可以这样自定义构造方法:
public record Person(String name, String address) {
public Person {
Objects.requireNonNull(name);
Objects.requireNonNull(address);
}
}
或者我们可以提供一些值包含默认值:
public record Person(String name, String address) {
public Person(String name) {
this(name, “Unknown”);
}
}
当然可以使用this.name来赋值也是一样的,但是由于所有字段默认都是final的,所以要求我们自定义的构造方法必须覆盖所有字段。
但是如果我们同时生命无参的构造,也重新覆盖定义全参数的构造,会导致编译报错,示例如下:
public record Person(String name, String address) {
public Person {
Objects.requireNonNull(name);
Objects.requireNonNull(address);
}
public Person(String name, String address) {
this.name = name;
this.address = address;
}
}
最后
现在其实从大厂招聘需求可见,在招聘要求上有高并发经验优先,包括很多朋友之前都是做传统行业或者外包项目,一直在小公司,技术搞的比较简单,没有怎么搞过分布式系统,但是现在互联网公司一般都是做分布式系统。
所以说,如果你想进大厂,想脱离传统行业,这些技术知识都是你必备的,下面自己手打了一份Java并发体系思维导图,希望对你有所帮助。
nNull(name);
Objects.requireNonNull(address);
}
public Person(String name, String address) {
this.name = name;
this.address = address;
}
}
最后
现在其实从大厂招聘需求可见,在招聘要求上有高并发经验优先,包括很多朋友之前都是做传统行业或者外包项目,一直在小公司,技术搞的比较简单,没有怎么搞过分布式系统,但是现在互联网公司一般都是做分布式系统。
所以说,如果你想进大厂,想脱离传统行业,这些技术知识都是你必备的,下面自己手打了一份Java并发体系思维导图,希望对你有所帮助。
[外链图片转存中…(img-wGcTSujy-1716449609919)]