早在2019年2月份,Java 语言架构师 Brian Goetz,曾写文抱怨“Java太啰嗦”或有太多的“繁文缛节”。他提到:开发人员想要创建纯数据载体类(plain data carriers)通常都必须编写大量低价值、重复的、容易出错的代码。如:构造函数、getter/setter、equals()、hashCode()以及toString()等。
以至于很多人选择使用IDE的功能来自动生成这些代码。还有一些开发会选择使用一些第三方类库,如Lombok等来生成这些方法。
JDK14中预览特性:神说要用record,于是就有了。实现一个简单的数据载体类,为了避免编写:构造函数,访问器,equals(),hashCode () ,toString ()等,Java 14推出record。
record 是一种全新的类型,它本质上是一个 final 类,同时所有的属性都是 final 修饰,它会自动编译出 public get 、hashcode 、equals、toString、构造器等结构,减少了代码编写量。
具体来说:当你用record 声明一个类时,该类将自动拥有以下功能:
注意: 不会生成setter,因为它是final 的
此外:
举例:
以前的写法:
class Point {private final int x;private final int y;Point(int x, int y) {this.x = x;this.y = y;}int x() {return x;}int y() {return y;}public boolean equals(Object o) {if (!(o instanceof Point)) return false;Point other = (Point) o;return other.x == x && other.y == y;}public int hashCode() {return Objects.hash(x, y);}@Overridepublic String toString() {return "Point{" +"x=" + x +", y=" + y +'}';}
}
新特性:
//会自动编译出 public get 、hashcode 、equals、toString、构造器等结构,减少了代码编写量。
// 但是没有setter,因为它是final 的
record Point(int x, int y) {// 这里可以商业不用写
}
/*** @author shkstart* @create 下午 6:20*/
public record Person(String name,Person partner) {//还可以声明静态的属性、静态的方法、构造器、实例方法public static String nation;public static String showNation(){return nation;}public Person(String name){this(name,null);}public String getNameInUpperCase(){return name.toUpperCase();}//不可以声明非静态的属性
// private int id;//报错
}//不可以将record定义的类声明为abstract的
//abstract record Order(){
//
//}//不可以给record定义的类声明显式的父类(非Record类)
//record Order() extends Thread{
//
//}
record的设计目标是提供一种将数据建模为数据的好方法。它也不是 JavaBeans 的直接替代品,因为record的方法不符合 JavaBeans 的 get 标准。另外 JavaBeans 通常是可变的,而记录是不可变的。尽管它们的用途有点像,但记录并不会以某种方式取代 JavaBean。