6

我正在使用 Java 15 的新记录功能,以及它如何与反射交互。我遇到了一些奇怪的行为,有时我可以通过反射访问记录的构造函数,有时不能。例如,给定以下 Java 文件:

Recording.java

public class Recording {
    public static void main(String[] args) {
        System.out.println("Constructors: " + MainRecord.class.getConstructors().length);
        System.out.println("Methods: " + MainRecord.class.getDeclaredMethods().length);
    }

    record MainRecord(int i, String s) {}
}

其行为如下:

❯ javac --enable-preview --release 15 Recording.java
Note: Recording.java uses preview language features.
Note: Recompile with -Xlint:preview for details.
❯ java --enable-preview Recording
Constructors: 0
Methods: 5

换句话说,调用getConstructors()没有找到任何构造函数(而调用 `getDeclaredMethods() 确实找到了方法)。我不明白为什么不,因为构造函数确实存在:

❯ javap Recording\$MainRecord
Compiled from "Recording.java"
final class Recording$MainRecord extends java.lang.Record {
  Recording$MainRecord(int, java.lang.String);
  public final java.lang.String toString();
  public final int hashCode();
  public final boolean equals(java.lang.Object);
  public int i();
  public java.lang.String s();
}

(将记录放在单独的 Java 文件中会得到相同的结果。)

但是,如果我从 JShell 做同样的事情:

❯ jshell --enable-preview
|  Welcome to JShell -- Version 15
|  For an introduction type: /help intro

jshell> record JShellRecord(int i, String s) {}
|  created record JShellRecord

jshell> JShellRecord.class.getConstructors().length
$2 ==> 1

所以,现在它确实找到了构造函数。

这是我正在使用的 Java 版本:

❯ java -version
openjdk version "15" 2020-09-15
OpenJDK Runtime Environment AdoptOpenJDK (build 15+36)
OpenJDK 64-Bit Server VM AdoptOpenJDK (build 15+36, mixed mode, sharing)

从 Java 14 编译和运行相同的程序确实有效:

❯ java -version
openjdk version "14.0.2" 2020-07-14
OpenJDK Runtime Environment AdoptOpenJDK (build 14.0.2+12)
OpenJDK 64-Bit Server VM AdoptOpenJDK (build 14.0.2+12, mixed mode, sharing)
❯ javac --enable-preview --release 14 Recording.java
Note: Recording.java uses preview language features.
Note: Recompile with -Xlint:preview for details.
❯ java --enable-preview Recording
Constructors: 1
Methods: 5

我知道在 Java 15 中,与 Java 14 相比,对记录的反射设置了许多限制,但如果我正确阅读JEP,这些限制仅适用于修改。寻找(也许是调用)构造函数似乎并不适用。

谁能告诉我这里发生了什么?我需要做什么才能通过反射在 Java 15 中查看记录的构造函数?

4

1 回答 1

10

getConstructors()仅返回public构造函数。用于getDeclaredConstructors()获取所有构造函数。

您的声明record MainRecord(int i, String s) {}缺少public修饰符,因此它创建了一个非public类和非public构造函数。参见JLS15 预览,§8.10.4

隐式声明的规范构造函数具有与记录类R相同的访问修饰符,除非记录类缺少访问修饰符,在这种情况下,规范构造函数具有包访问权限

这确实与 JDK 14 预览版不同。JDK 15 预览文档的开头说:

这些更改与 Java SE 14 中 Records 的第一个预览版中的相同,除了以下内容:

…</p>

  • 8.10.4 删除了规范构造函数必须是公共的要求。任何访问修饰符必须至少提供与记录类一样多的访问权限。如果隐式声明规范构造函数,则其访问修饰符与记录类相同。

似乎,在 JShell 中创建的顶级类是隐式的public

> jdk-15\bin\jshell --enable-preview
|  Welcome to JShell -- Version 15
|  For an introduction type: /help intro

jshell> record JShellRecord(int i, String s) {}
|  created record JShellRecord

jshell> JShellRecord.class.getConstructors()[0]
$2 ==> public JShellRecord(int,java.lang.String)

jshell> java.lang.reflect.Modifier.isPublic(JShellRecord.class.getModifiers())
$3 ==> true

jshell>
于 2020-09-21T14:38:07.620 回答