我设计了一个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)
我发现原因是在控制器类的initialize方法中,我不能在任何静态组件中使用内置方法。(例如:staticMyTextField.setText()在java 8中导致了这个问题,但在java 7中没有)。我在javafx指南中找不到任何关于这个问题的文档。有人能提供一些关于为什么这会在Java 8中导致问题的想法吗?如果有的话,也可以分享与此相关的文档。
1条答案
按热度按时间5lhxktic1#
这听起来像是在尝试将
TextField
注入静态字段。这显然在JavaFX 2.2中起作用,但在JavaFX 8中不起作用。由于没有官方文档支持这种用法,因此它并没有真正违反向后兼容性,尽管公平地说,关于
FXMLLoader
的文档确实很糟糕。将
@FXML
注入的字段设置为静态的实际上没有多大意义。它为FXML文件中的每个元素创建新对象。新控制器示例与对FXMLLoader.load(...)
的每次调用相关联,并且该控制器 * 示例 * 中的字段被注入为FXML元素创建的对应对象。因此注入的字段必须特定于控制器示例。如果您在控制器中有一个静态注入字段,并且您加载了同一个FXML文件两次并在UI中显示两次,那么您将无法引用两组控件。从FXML引用的
@FXML
标记的静态方法也将不起作用。例如:
以及:
这将失败,并出现以下情况:
要解决这个问题,只需从方法规范中删除static修饰符。
更新:回复评论中的问题
特别是,不要使用静态字段只是为了使它们能够从类外部访问。静态字段具有属于类的单个值,而不是类的每个示例的值,并且只有在有意义的情况下才应该决定将字段静态化。换句话说,
static
定义了 scope,而不是 accessibility。为了允许访问示例数据,你只需要有一个对示例的引用。FXMLLoader
有一个getController()
方法,允许你检索对控制器的引用。相关的一点:从控制器中公开UI控件也不是一个好主意。您应该公开数据。例如,而不是在控制器中定义
getTextField()
方法,而是定义一个textProperty()
方法,该方法返回表示TextField
内容的StringProperty
。这样做的原因是,当您的老板来到办公室并告诉您他希望将TextField
替换为TextArea
或ComboBox<String>
或其他控件时,那么如果控制器之外的类正在使用你的TextField
的话,这将是一件非常困难的事情。你的控制器所表示的数据的结构比数据如何呈现给用户的实现更不可能改变。对于一些例子