[java] 编译之后的记录类(Record Classes)长什么样子(上)

背景

JEP 395: Records 中提到 JDK 16 \text{JDK 16} JDK 16 正式支持记录类( Record Classes \text{Record Classes} Record Classes)。那么记录类在 class \text{class} class 文件中长什么样子呢?让我们一起来探索吧。

要点

  • 规范构造函数( canonical constructor \text{canonical constructor} canonical constructor)
    • 如果我们在记录类中没有定义构造函数,那么记录类中会出现隐式的 canonical constructor \text{canonical constructor} canonical constructor
  • 对每个 Record Component \text{Record Component} Record Component 而言
    • 记录类中会有一个与之对应的 private \text{private} private, final \text{final} final, 非 static \text{static} static 字段
    • 记录类中会有一个与之对应的被称为 accessor method \text{accessor method} accessor method 的方法

正文

例子

我从 JEP 395: Records 里找了个例子 ⬇️ (略有改动)

java 复制代码
public record Point(int x, int y) {
}

我们将上述代码保存为 Point.java \text{Point.java} Point.java。下方的命令可以编译 Point.java \text{Point.java} Point.java (请注意,所使用的 javac \text{javac} javac 版本应当高于或者等于 16 \text{16} 16)

bash 复制代码
javac Point.java

编译后,会得到 Point.class \text{Point.class} Point.class 文件。使用如下命令可以查看 Point.class \text{Point.class} Point.class 的内容

bash 复制代码
javap -p Point

这个命令在我电脑上运行的结果如下 ⬇️

text 复制代码
Compiled from "Point.java"
public final class Point extends java.lang.Record {
  private final int x;
  private final int y;
  public Point(int, int);
  public final java.lang.String toString();
  public final int hashCode();
  public final boolean equals(java.lang.Object);
  public int x();
  public int y();
}

我画了对应的类图 ⬇️

在反编译的结果中,我们看到了

  • Point \text{Point} Point 的构造函数
  • x x x 字段和 y y y 字段
  • x ( ) x() x() 方法和 y ( ) y() y() 方法

但是我们在 Point.java \text{Point.java} Point.java 中 并没有 显式定义这些内容,那么它们是从哪里来的呢?

规范构造函数(Canonical Constructor

如果我们在记录类没有显式定义构造函数,那么记录类中会出现隐式的 canonical constructor \text{canonical constructor} canonical constructor。 8.10.4. Record Constructor Declarations 提到了相关细节 ⬇️

可以用如下命令查看 Point.class \text{Point.class} Point.class 的详细内容

bash 复制代码
javap -v -p Point

完整的结果比较长,其中和构造函数相关的部分如下

text 复制代码
  public Point(int, int);
    descriptor: (II)V
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=2, locals=3, args_size=3
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Record."<init>":()V
         4: aload_0
         5: iload_1
         6: putfield      #7                  // Field x:I
         9: aload_0
        10: iload_2
        11: putfield      #13                 // Field y:I
        14: return
      LineNumberTable:
        line 1: 0
    MethodParameters:
      Name                           Flags
      x
      y

我们可以手动对其进行反编译 ⬇️

java 复制代码
// 以下内容是我手动反编译的结果,仅供参考
public Record(int x, int y) {
    super();
    this.x = x;
    this.y = y;
}

Record Components

8.10. Record Classes 中提到了 Record Components \text{Record Components} Record Components ⬇️ (重要的部分我用绿色框和绿色箭头标了出来)

Point \text{Point} Point 这个记录类而言,以下两者都是它的 Record Component \text{Record Component} Record Component

  • int x \text{int x} int x
  • int y \text{int y} int y

8.10.3. Record Members 提到了如下内容 ⬇️ (重要的部分我用绿色框标了出来)

对每个 Record Component \text{Record Component} Record Component 而言,

  • 记录类中会有一个与之对应的 private \text{private} private, final \text{final} final, 非 static \text{static} static 字段
  • 记录类中会有一个与之对应的被称为 accessor method \text{accessor method} accessor method 的方法

我们可以用如下的命令查看 Point.class \text{Point.class} Point.class 的详细内容

bash 复制代码
javap -v -p Point

完整的结果比较长,其中和 Record Components \text{Record Components} Record Components 相关的部分如下

text 复制代码
  private final int x;
    descriptor: I
    flags: (0x0012) ACC_PRIVATE, ACC_FINAL

  private final int y;
    descriptor: I
    flags: (0x0012) ACC_PRIVATE, ACC_FINAL
    
  public int x();
    descriptor: ()I
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: getfield      #7                  // Field x:I
         4: ireturn
      LineNumberTable:
        line 1: 0

  public int y();
    descriptor: ()I
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: getfield      #13                 // Field y:I
         4: ireturn
      LineNumberTable:
        line 1: 0

我们可以手动对其进行反编译 ⬇️

java 复制代码
// 以下内容是我手动反编译的结果,仅供参考
private final int x;
private final int y;

public int x() {
    return this.x;
}

public int y() {
    return this.y;
}

参考资料

其他

Point 的类图所用到的代码

puml 复制代码
@startuml
'https://plantuml.com/class-diagram

@startuml

title <i>Point</i> 的类图
caption \n\n
' caption 的内容只是为了防止掘金平台生成的水印遮盖图中的文字

abstract java.lang.Record
class Point
java.lang.Record <-- Point

abstract java.lang.Record {
    # Record()
    + {abstract} boolean equals(Object)
    + {abstract} int hashCode()
    + {abstract} String toString()
}

class Point {
  - final int x
  - final int y
  + Point(int, int)
  + final String toString()
  + final int hashCode()
  + final boolean equals(Object)
  + int x()
  + int y()
}

note left of java.lang.Record::equals
override 了 <i>java.lang.Object</i> 中的方法
end note

note left of java.lang.Record::hashCode
override 了 <i>java.lang.Object</i> 中的方法
end note

note left of java.lang.Record::toString
override 了 <i>java.lang.Object</i> 中的方法
end note

@enduml
相关推荐
Zella折耳根7 分钟前
复习篇-常用实用类
java
devilnumber6 小时前
Java 递归算法 详解 + 核心要点 + 实战运用 + 避坑指南
java·开发语言·算法
独泪了无痕6 小时前
MyBatis魔法堂:结果集映射
后端·mybatis
copyer_xyf7 小时前
LangChain 调用 LLM
后端·python·agent
copyer_xyf7 小时前
Prompt 组织管理
后端·python·agent
asdfg12589637 小时前
JavaBean是什么?怎么理解?有什么用途?
java·开发语言
摇滚侠9 小时前
SpringMVC 入门到实战 文件上传 75-77
java·后端·spring·maven·intellij-idea
GIS数据转换器9 小时前
城市排水生命线安全运行监测平台深度解析
java·运维·人工智能·python·安全·数据挖掘·无人机
华如锦10 小时前
面了很多 Java转AI Agent方向,一些面试题总结
java·开发语言·人工智能·python·ai
睡不醒男孩03082310 小时前
CLup 6.x 版本中针对StarRocks 存算一体集群的完整操作手册
java·服务器·网络·clup