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

2022-09-02 20:07:11

我设计了一个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)

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


答案 1

这听起来像是你试图将 a 注入到静态字段中。类似的东西TextField

@FXML
private static TextField myTextField ;

这显然在JavaFX 2.2中起作用。它在JavaFX 8中不起作用。由于没有官方文档支持这种用法,因此它并没有真正违反向后兼容性,尽管公平地说,关于它到底做什么的文档非常糟糕。FXMLLoader

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

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

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

相关的一点是:从控制器公开UI控件也不是一个好主意。您应改为公开数据。例如,不是在控制器中定义方法,而是定义一个返回表示 .这样做的原因是,当你的老板来到办公室并告诉你他希望被一个,或一个,或者其他一些控件所取代时,那么如果控制器以外的类使用你的.of,那将会困难得多。与实现如何向用户呈现数据相比,控制器所表示的数据结构发生更改的可能性要小得多。getTextField()textProperty()StringPropertyTextFieldTextFieldTextAreaComboBox<String>TextField

有关某些示例


答案 2

推荐