javafx 8 兼容性问题 - FXML 静态字段或方法

标签 java javafx-2 java-7 java-8 javafx-8

我设计了一个 javafx 应用程序,它在 jdk 7 中运行良好。当我尝试在 java 8 中运行它时,出现以下异常:

javafx.fxml.LoadException: 
at javafx.fxml.FXMLLoader.constructLoadException(FXMLLoader.java:2617)
    at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2595)
    at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:2441)
    at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3230)
    at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3191)
    at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3164)
    at javafx.fxml.FXMLLoader.loadImpl(FXMLLoader.java:3140)
    at javafx.fxml.FXMLLoader.load(FXMLLoader.java:3132)


Exception in thread "JavaFX Application Thread" java.lang.NullPointerException: Root cannot be null
    at javafx.scene.Scene.<init>(Scene.java:364)
    at javafx.scene.Scene.<init>(Scene.java:232)
        at com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:86)
    at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:238)
    at com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:191)
    at com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:58)
    at com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
    at com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:74)
    at com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:54)
    at javafx.event.Event.fireEvent(Event.java:204)
    at javafx.concurrent.EventHelper.fireEvent(EventHelper.java:219)
    at javafx.concurrent.Task.fireEvent(Task.java:1357)
    at javafx.concurrent.Task.setState(Task.java:720)
    at javafx.concurrent.Task$TaskCallable$2.run(Task.java:1438)
    at com.sun.javafx.application.PlatformImpl$6$1.run(PlatformImpl.java:301)
    at com.sun.javafx.application.PlatformImpl$6$1.run(PlatformImpl.java:298)
    at java.security.AccessController.doPrivileged(Native Method)
    at com.sun.javafx.application.PlatformImpl$6.run(PlatformImpl.java:298)
    at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95)
    at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
    at com.sun.glass.ui.win.WinApplication.access$300(WinApplication.java:39)
    at com.sun.glass.ui.win.WinApplication$4$1.run(WinApplication.java:112)
    at java.lang.Thread.run(Thread.java:744)

我发现原因是在 Controller 类的初始化方法中,我无法在任何静态组件中使用内置方法。 (例如: staticMyTextField.setText() 在 java 8 中导致问题,但在 java 7 中则不会)。我无法在 javafx 指南中找到任何与此相关的记录。有人可以提供一些关于为什么这会在 Java 8 中引起问题的想法吗?并分享与此相关的文档(如果有)。

最佳答案

听起来您正在尝试注入(inject) TextField进入静态场。类似的东西

@FXML
private static TextField myTextField ;

这显然在 JavaFX 2.2 中有效。它在 JavaFX 8 中不起作用。由于没有官方文档支持这种用法,因此它并没有真正违反向后兼容性,尽管公平地说,文档说明了 FXMLLoader 的具体内容。确实很糟糕。

创建 @FXML 并没有多大意义。 -注入(inject)字段静态。当您加载 FXML 文件时,它会为 FXML 文件中的每个元素创建新对象。每次调用FXMLLoader.load(...)都会关联一个新的 Controller 实例。并且该 Controller 实例中的字段将注入(inject)为 FXML 元素创建的相应对象。因此注入(inject)的字段必须特定于 Controller 实例。如果 Controller 中有静态注入(inject)字段,并且加载相同的 FXML 文件两次并在 UI 中显示两次,那么您将无法引用两组控件。

<小时/>

@FXML从 FXML 引用的标记静态方法也将不起作用。

例如,使用:

@FXML
private static void changeUser(ActionEvent e) {
    // System.out.println("test");
}

还有:

<Button text="Change User" onAction="#changeUser"/>

这将失败:

javafx.fxml.LoadException: Error resolving onAction='#changeUser', either the event handler is not in the Namespace or there is an error in the script.

要解决此问题,只需从方法规范中删除 static 修饰符即可。

<小时/>

更新:回复评论中的问题

特别是,不要仅仅为了能够从类外部访问静态字段而使用静态字段。静态字段具有属于该类的单个值,而不是该类的每个实例的值,并且只有在有意义的情况下才应做出将字段设为静态的决定。换句话说,static定义范围,而不是可访问性。要允许访问实例数据,您只需拥有对该实例的引用即可。 FXMLLoader有一个getController()方法允许您检索对 Controller 的引用。

相关点:从 Controller 公开 UI 控件也不是一个好主意。相反,您应该公开数据。例如,不要定义 getTextField() Controller 中的方法,而是定义 textProperty()返回 StringProperty 的方法代表 TextField 的内容。原因是当你的老板来到办公室并告诉你他想要TextField时替换为 TextArea ,或 ComboBox<String> ,或其他一些控件,那么如果 Controller 外部的类正在使用您的 TextField ,那么这将会变得更加困难。 。与向用户呈现数据的方式相比, Controller 所表示的数据结构发生变化的可能性要小得多。

举一些例子

关于javafx 8 兼容性问题 - FXML 静态字段或方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57679882/

相关文章:

java - 对 C/C++ 对象和 C/C++ 对象对 Java/Objective C 的 JSON 响应

Java嵌套循环: Indent text

java - 在 Java 中是否有调用方法的对象的名称?

java - JavaFX 8 是否要实现文本字段验证支持?

java - 在 Jackson 的单个文件中反序列化来自多个 YAML 文档的 POJO

javaFX - 如何创建可调整大小的矩形?

java - Maven 构建失败,如何在 Eclipse 中将 -source 1.5 更改为 -source 7 或更高版本

java - JavaFX 图表上的 Swing 上下文菜单

tomcat7 - Tomcat 7 现在是否与 Java 7 兼容?

java - 如何对我现有的 javafx 应用程序进行多线程处理