一尘不染

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

java

我设计了一个在jdk 7中正常工作的javafx应用程序。当我尝试在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)

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


阅读 717

收藏
2020-02-25

共1个答案

一尘不染

听起来你正在尝试将a TextField注入静态字段。就像是

@FXML
private static TextField myTextField ;

这显然在JavaFX 2.2中有效。它在JavaFX 8中不起作用。由于没有官方文档曾经支持过这种用法,所以实际上并没有违反向后兼容性,尽管公平地说,关于这样做的文档FXMLLoader确实很糟糕。

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

更新:对评论中问题的回答

特别是,不要仅使用静态字段来使它们可以从类外部进行访问。静态字段只有一个属于该类的值,而不是该类的每个实例的值,并且只有在有意义的情况下,才应决定使字段成为静态。换句话说,static定义范围,而不是可访问性。要允许访问实例数据,你只需要引用实例即可。使用FXMLLoadergetController()方法可以检索对控制器的引用。

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

2020-02-25