我正在学习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,这些限制只适用于修改。查找(也许调用(构造函数似乎并不适用。
有人能告诉我这里发生了什么吗?我需要做什么才能通过反射在Java15中看到记录的构造函数?
getConstructors()
仅返回public
构造函数。使用getDeclaredConstructors()
获取所有构造函数。
您的声明record MainRecord(int i, String s) {}
缺少public
修饰符,因此它创建了一个非public
类和非public
构造函数。参见JLS15预览,§8.10.4
隐式声明的规范构造函数与记录类R具有相同的访问修饰符,除非记录类缺少访问修饰符,在这种情况下,规范构造函数具有包访问
这确实与JDK14预览版有所不同。JDK15预览文档的开头写道:
这些更改与Java SE 14中Records的第一次预览中的更改相同,除了以下内容:
…
- 8.10.4删除了规范构造函数必须是公共的要求。任何访问修饰符都必须提供至少与记录类一样多的访问权限。如果规范构造函数是隐式声明的,那么它的访问修饰符与记录类相同
在JShell中创建的顶级类似乎是隐式的public
。
> jdk-15binjshell --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>