我设计了一个在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问题的想法吗?并且还共享与此相关的文档。
答案 0 :(得分:34)
听起来你正试图将TextField
注入静态字段。像
@FXML
private static TextField myTextField ;
这显然适用于JavaFX 2.2。它在JavaFX 8中不起作用。由于没有官方文档曾经支持过这种用法,所以它并没有真正违反向后兼容性,尽管公平地说明FXMLLoader
所做的文件非常糟糕。
使@FXML
- 注入的字段静态没有多大意义。加载FXML文件时,它会为FXML文件中的每个元素创建新对象。每个对FXMLLoader.load(...)
的调用都会关联一个新的控制器实例,并且该控制器实例中的字段将注入为FXML元素创建的相应对象。因此,注入的字段必须特定于控制器实例。如果你在控制器中有一个静态注入字段,并且你加载了相同的FXML文件两次并在UI中显示两次,那么你将无法引用这两组控件。
更新:回复评论中的问题
特别是,不要仅使用静态字段来使它们可以从类外部访问。静态字段具有属于该类的单个值,而不是该类的每个实例的值,并且只有在有意义的情况下才应该将字段设置为静态。换句话说,static
定义范围,而非辅助功能。要允许访问实例数据,您只需要引用该实例。 FXMLLoader
具有getController()
方法,允许您检索对控制器的引用。
一个相关的观点:从控制器公开UI控件也不是一个好主意。您应该改为公开数据。例如,不是在控制器中定义getTextField()
方法,而是定义一个textProperty()
方法,该方法返回表示StringProperty
内容的TextField
。这样做的原因是,当你的老板来到办公室并告诉你他希望TextField
被TextArea
,ComboBox<String>
或其他控件替换时,那就是如果控制器外的类正在使用你的TextField
,那将会变得更加困难。控制器所代表的数据结构的变化远不如实现数据呈现给用户的方式。
举个例子