前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java 8之后的那些新特性(六):记录类 Record Class

Java 8之后的那些新特性(六):记录类 Record Class

作者头像
御剑
发布2022-06-07 19:13:33
2.6K0
发布2022-06-07 19:13:33
举报
文章被收录于专栏:微言码道

Java是一门面向对象的语言,而对于面向对象的语言中,一个众所周知的概念就是,对象是包含属性与行为的。

比如HR系统中都会有雇员的概念,那雇员会有姓名,ID身份,性别等,这些我们称之为属性;而雇员同时肯定会有入职,离职,薪金被调整等业务上的 操作,这些我们称之为行为。

所以,在面向对象的语言中,一个映射业务概念的对象,是应该包含属性以及行为,这样才是完整的面向对象的。

但这并不代表全部,在实现的编码过程中,我们会经常遇到一些类,它更多的只是一种数据载体。比如服务间的数据交互,REST API的承载对象等,它可能只是技术上单纯用来做 数据交互或承担数据传输任务,这样的类中其实并不需要太多方法。

这样的类,我们可以称之为数据类,在Java这门语言中,它以不同的概念或形式出现,比如DTO对象,VO对象,或POJO等。而在过往,Java语言中处理类似的类是非常麻烦的。

但这一切,在Java引入Record Class的概念后,就简化很多了。

这周,我继续和大家聊一聊Java 8之后的那些新特性。这一次我来讲下记录类 Record Class

这是Java 8之后的那些新特性系列的第五篇,这个系列的其它文章是:

  1. 1. Java 8之后的那些新特性 (一) :局部变量var
  2. 2. Java 8之后的那些新特性 (二) :文本块 Text Blocks
  3. 3. Java 8之后的那些新特性 (三) :Java System Logger
  4. 4. Java 8之后的那些新特性 (四) :网络请求 Java Http Client
  5. 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. 1. 数据类的基本属性
  2. 2. 属性的getter,setter方法
  3. 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,仍然有一些基本原则你需要知道。

不能在record类的body中添加属性,属性只能定义在类的括号后面(称为header)

代码语言:javascript
复制
public record EmployeeDTO(String name,String idCard,int age){
    //这是不允许的
    private String description;
    
}

可以在record类中添加静态属性与方法

代码语言:javascript
复制
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);
        //...
    }

这个在一些局部方法中需要封装一些参数时,又没必要把这个类定义在外面时非常有用。

Java中的所有record类,都默认实现了Record接口

代码语言:javascript
复制
    @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)以了解我在维护的全栈式领域驱动开源框架。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2022-04-18,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 微言码道 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 啰嗦的数据类
  • Kotlin的data class
  • Java Record 记录类
  • record class需要关注的点
  • 总结
  • 关于我
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档