我想知道JavaFX 8中@NamedArg注释的用例是什么
javadoc没有给我们更多细节, Javadoc:提供有关参数名称信息的注释。
互联网上没有更多信息,文档和示例。
也许有人可以提供帮助?
问候。
答案 0 :(得分:47)
@NamedArg
注释允许FXMLLoader
实例化一个没有零参数构造函数的类。
技术背景:
FXMLLoader
使用反射创建对象。通常,如果您使用与不带参数的构造函数对应的类的标记,则通过调用Class.newInstance()
从该类创建一个对象,该对象将调用无参数构造函数。
如果仅使用带参数的构造函数定义类,则这是有问题的。主要问题是Java语言规范不要求在运行时保留参数(到方法或构造函数)的名称。这意味着FXMLLoader
没有直接,有保证的方式来确定哪个参数具有给定名称。
为了使这一点具体化,假设我们按如下方式定义Person
类:
package application;
import javafx.beans.NamedArg;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
public class Person {
private final StringProperty firstName ;
private final StringProperty lastName ;
public Person(String firstName, String lastName) {
this.firstName = new SimpleStringProperty(this, "firstName", firstName);
this.lastName = new SimpleStringProperty(this, "lastName", lastName);
}
// methods....
}
在FXML中,我们可能会尝试按如下方式创建Person
:
<Person firstName="Jacob" lastName="Smith"/>
这不起作用,因为FXML加载器无法保证Person
类的运行时表示保留关于哪个构造函数参数为firstName
以及哪个为lastName
的信息
历史背景
Java 2.2定义了与每个控件对应的“Builder”类。这些构建器类遵循标准构建器模式。当FXMLLoader
遇到引用没有零参数构造函数的类的标记时,它将使用相应的构建器来创建实例。
不幸的是,构建器类的实现存在缺陷,they were deprecated in JavaFX 8,将在更高版本(可能是JavaFX 9)中删除。这给FXMLLoader
留下了一个问题,它不再需要依赖构建器类来实例化没有零参数构造函数的类。一个真实的例子是Color
类,它没有零参数构造函数,并且将删除它的构建器类。
@NamedArgs
对此的修复是引入一个注释,用于在运行时保留方法(或构造函数)参数的名称。通过反射,我们可以查询构造函数/方法的参数列表,并获取每个参数的类型(但不是名称)。也可以查询每个参数的任何注释,并获取这些注释的值。因此,@NamedArg
注释是专门为了在运行时保留参数名称而引入的。
示例强>
例如,使用我们上面介绍的Person
类:
package application;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
public class Person {
private final StringProperty firstName ;
private final StringProperty lastName ;
public Person(String firstName, String lastName) {
this.firstName = new SimpleStringProperty(this, "firstName", firstName);
this.lastName = new SimpleStringProperty(this, "lastName", lastName);
}
public final StringProperty firstNameProperty() { return firstName; }
public final String getFirstName() { return firstNameProperty().get(); }
public final void setFirstName(final String firstName) { firstNameProperty().set(firstName); }
public final StringProperty lastNameProperty() { return lastName; }
public final String getLastName() { return lastNameProperty().get(); }
public final void setLastName(final String lastName) { lastNameProperty().set(lastName); }
}
如果您尝试使用FXML加载:
Person.fxml:
<?xml version="1.0" encoding="UTF-8"?>
<?import application.Person?>
<Person firstName="Jacob" lastName="Smith" xmlns:fx="http://javafx.com/fxml/1" />
Main.java:
package application;
import java.io.IOException;
import javafx.fxml.FXMLLoader;
public class Main {
public static void main(String[] args) throws IOException {
Person person = FXMLLoader.load(Main.class.getResource("Person.fxml"));
System.out.println(person.getFirstName()+" "+person.getLastName());
}
}
然后您在运行时看到错误:
Caused by: java.lang.NoSuchMethodException: application.Person.<init>()
表示FXMLLoader
正在寻找不带参数的构造函数(Person.<init>()
)。
在JavaFX 8中,您可以通过使用@NamedArg
注释指定参数的名称来解决问题:
package application;
import javafx.beans.NamedArg;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
public class Person {
private final StringProperty firstName ;
private final StringProperty lastName ;
public Person(@NamedArg("firstName") String firstName, @NamedArg("lastName") String lastName) {
this.firstName = new SimpleStringProperty(this, "firstName", firstName);
this.lastName = new SimpleStringProperty(this, "lastName", lastName);
}
public final StringProperty firstNameProperty() { return firstName; }
public final String getFirstName() { return firstNameProperty().get(); }
public final void setFirstName(final String firstName) { firstNameProperty().set(firstName); }
public final StringProperty lastNameProperty() { return lastName; }
public final String getLastName() { return lastNameProperty().get(); }
public final void setLastName(final String lastName) { lastNameProperty().set(lastName); }
}
这将允许FXMLLoader
根据需要加载课程。
请注意,您还可以通过定义构建器类来解决此问题,并且这也适用于JavaFX 2.0及更高版本。 JavaFX团队决定(可能是正确的)使用这种方法的方式不会受到构建器初始实现中存在的错误的影响,这会给框架的代码库增加太多的膨胀。
package application;
public class PersonBuilder {
private String firstName ;
private String lastName ;
private PersonBuilder() { }
public static PersonBuilder create() {
return new PersonBuilder();
}
public PersonBuilder firstName(String firstName) {
this.firstName = firstName ;
return this ;
}
public PersonBuilder lastName(String lastName) {
this.lastName = lastName ;
return this ;
}
public Person build() {
return new Person(firstName, lastName);
}
}
显然,如果您使用的是JavaFX 8,那么构造函数注释方法的工作要少得多。
<强>参考文献:强>