Java是一门面向对象的语言,而对于面向对象的语言中,一个众所周知的概念就是,对象是包含属性与行为的。
比如HR系统中都会有雇员的概念,那雇员会有姓名,ID身份,性别等,这些我们称之为属性;而雇员同时肯定会有入职,离职,薪金被调整等业务上的 操作,这些我们称之为行为。
所以,在面向对象的语言中,一个映射业务概念的对象,是应该包含属性以及行为,这样才是完整的面向对象的。
但这并不代表全部,在实现的编码过程中,我们会经常遇到一些类,它更多的只是一种数据载体。比如服务间的数据交互,REST API的承载对象等,它可能只是技术上单纯用来做 数据交互或承担数据传输任务,这样的类中其实并不需要太多方法。
这样的类,我们可以称之为数据类,在Java这门语言中,它以不同的概念或形式出现,比如DTO对象,VO对象,或POJO等。而在过往,Java语言中处理类似的类是非常麻烦的。
但这一切,在Java引入Record Class的概念后,就简化很多了。
这周,我继续和大家聊一聊Java 8之后的那些新特性。这一次我来讲下记录类 Record Class
这是Java 8之后的那些新特性系列的第五篇,这个系列的其它文章是:
- 1. Java 8之后的那些新特性 (一) :局部变量var
- 2. Java 8之后的那些新特性 (二) :文本块 Text Blocks
- 3. Java 8之后的那些新特性 (三) :Java System Logger
- 4. Java 8之后的那些新特性 (四) :网络请求 Java Http Client
- 5. Java 8之后的那些新特性 (五) :Helpful NullPointerExceptions
啰嗦的数据类
如果你在Java的代码项目中,或多或少一定会接触这些类的概念
- • DTO (data transfer object) 数据传输对象
- • VO (Value Object) 值对象
- • POJO (Plain Old Java Object) 普通Java旧对象
上面这些概念可能在不同的框架,不同的项目中都可能出现,但它们基本上都代表一个含义,就是
类只包含基本的属性与getter,setter方法,不存在业务上的方法,主要是做为数据传输的载体类
这一类的,我把它统称为数据类
而在过往,Java中定义这样的数据类基本是这样的,以如下代码为示例:
代码语言:javascript复制public class EmployeeDTO {
private String name;
private String idCard;
private int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getIdCard() {
return idCard;
}
public void setIdCard(String idCard) {
this.idCard = idCard;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
EmployeeDTO that = (EmployeeDTO) o;
return age == that.age && name.equals(that.name) && idCard.equals(that.idCard);
}
@Override
public int hashCode() {
return Objects.hash(name, idCard, age);
}
@Override
public String toString() {
return "EmployeeDTO{"
"name='" name '''
", idCard='" idCard '''
", age=" age
'}';
}
}
这样的对象,基本包含以下基本要素:
- 1. 数据类的基本属性
- 2. 属性的getter,setter方法
- 3. 类的hashCode,equals以及toString方法
你一定有编写过类似Java类的经历,这些类的编写实质上非常啰嗦。其实都是大同小异的。
由于这些重复啰嗦的东西非常令人讨厌,以至于Java生态中出现了一个解决这个问题的框架,就是java lombok,lombok就是简化很多Java类编写的代码的一个侵入式的框架。
lombok事实上还非常流行,应该很多Java程序员都用过。它的代码大致是这样:
代码语言:javascript复制@Getter
@Setter
@NoArgsConstructor
@EqualsAndHashCode
public class EmployeeDTO {
private String name;
private String idCard;
private int age;
}
当然,lombok这框架也提供了更多的类似支持Builder模式的功能。
我在数年之前并不清楚这个玩意,直到有一次查看公司另一个项目的代码时,第一次见到类似的玩意。当时还非常困惑,因为我不清楚是怎么回事。
其实我个人并不主张用这样的框架,最重要的原因是它的侵入性太强。
当然,从这一点上也可以感受到,大家是多讨厌Java中这种重复啰嗦的定义。
Kotlin的data class
还是来参考下友军是怎么做的吧。Kotlin这门语言,号称better java,确实是事实。在Kotlin语言中,语言设计上就完全避开了这一点。
Kotlin中有一个Data Class的概念,它就是用来解决这个问题的。
代码如下:
代码语言:javascript复制data class EmployeeDTO(val name:String,val idCard:String,val age:int)
在Kotlin中,你可以定义data class,当你定义一个data class时,编译器会自动帮你
- • 生成hasCode以及equals方法
- • toString方法
而getter,setter方法在Kotlin中本来就是默认不需要显式定义的,编译器帮你自动作了,这是针对所有类都有的行为。
所以,当我们以Kotlin的data class来对比Java中定义一个数据录时,其简洁性确实提升了几个级别。
不过,好在,Java语言并未停止进步,它在Java 14,15版本中引入了预览版的Record Class特性,并在Java 17中将其正式引入。
Java Record 记录类
大致说来,除了Kotlin中叫data class,Java中叫Record Class这个名称不太一样以外,其它的都是极为类似的。
我们用Java 17中的Record Class 来重写上述这个类,代码是这样的:
代码语言:javascript复制public record EmployeeDTO(String name,String idCard,int age){}
是不是几乎和Kotlin中的data class一模一样呢?
是的,就是这么回事,它简化了数据类的定义。所以如果你非常厌烦Java数据类的重复定义,与其去使用lombok这种侵入性非常强的第三方库,还不如升级使用
Java 17。
record class需要关注的点
当然,关于record class,仍然有一些基本原则你需要知道。
代码语言:javascript复制不能在record类的body中添加属性,属性只能定义在类的括号后面(称为header)
public record EmployeeDTO(String name,String idCard,int age){
//这是不允许的
private String description;
}
代码语言:javascript复制可以在record类中添加静态属性与方法
public record EmployeeDTO(String name,String idCard,int age){
//这是允许的
private static System.Logger logger = System.getLogger(EmployeeDTO.class.getName());
}
代码语言:javascript复制可以添加额外的类方法,这是允许的
public record EmployeeDTO(String name,String idCard,int age){
//这是允许的
public String toJson(){
//...
return "";
}
}
比如,你可以添加一个方法,有时候我们需要将数据对象转换为JSON来传输或存储,那就添加一个toJson方法就好了.
代码语言:javascript复制可以覆盖默认生成的一些东西
public record EmployeeDTO(String name,String idCard,int age){
//这是允许的
public int getAge(){
return age;
}
}
可以在方法中定义Local Record Classes
在方法内部,你可以定义一个局部本地的record类
代码语言:javascript复制 public void calculateLocation(double x,double y){
//定义一个本地record类
record Point(double x, double y) {}
var point = new Point(x,y);
//...
}
这个在一些局部方法中需要封装一些参数时,又没必要把这个类定义在外面时非常有用。
代码语言:javascript复制Java中的所有record类,都默认实现了Record接口
@Test
void testRecord(){
record Point(double x,double y){}
var point = new Point(0,0);
//Point是实现了Record接口的
Assertions.assertTrue(point instanceof Record);
}
总结
好了,现在你知道record class是怎么一回事了吧,它确实是非常有价值的一个新特性了。如果你使用的是JDK 17,或是JDK 14,15,都是可以用上这个特性的。
当然,关于我不主张使用lombok这样的框架,只是个人之见了,有机会我可以再聊下我的想法。
下周我们继续,聊一聊Java 8之后的那些新特性,还有挺多值得说的一些特性了。
关于我
我是御剑,一个致力于实践与传播编码之道的全栈式程序员。
访问微言码道
(https://taoofcoding.tech)以阅读更多我写的文章;
访问myddd
(https://myddd.org)以了解我在维护的全栈式领域驱动开源框架。