JavaFX8兼容性问题-FXML静态字段

JavaFX8兼容性问题-FXML静态字段,java,javafx-2,java-7,java-8,javafx-8,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.load

我设计了一个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指南中找到任何与此相关的文档。有人能提供一些想法,解释为什么这会导致Java8出现问题吗?如果有的话,还可以共享与此相关的文档。

这听起来像是在试图将文本字段注入静态字段。差不多

@FXML
private static TextField myTextField ;
这显然在JavaFX2.2中起作用。它在JavaFX8中不起作用。因为没有官方文档支持这种使用,所以它并没有真正违反向后兼容性,尽管公平地说,关于FXMLLoader的确切功能的文档非常糟糕

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

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

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

相关的一点:从控制器公开UI控件也不是一个好主意。您应该公开数据。例如,与其在控制器中定义getTextField方法,不如定义一个textProperty方法,该方法返回表示TextField内容的StringProperty。这样做的原因是,当你的老板来到办公室,告诉你他想用TextArea、ComboBox或其他控件替换TextField时,如果控制器之外的类正在使用你的TextField,情况会变得更加困难。控制器所表示的数据的结构与向用户呈现数据的方式相比,变化的可能性要小得多

例如


你能提供代码示例吗?请参考此链接。这在JDK 7中工作,但在JDK 8中不工作。请提供帮助?谢谢James。是否仍然可以从其他控制器类访问组件的内容?例如,我在一个控制器类的listView中显示了一些内容复选框和标签。如果我想从另一个控制器类访问它而不改变或改变它的内容,我应该怎么做?+1作为一个很好的解释。我一直认为在这种情况下,静态是有意义的。您可以尽可能晚地使用static。这是到目前为止我在这方面看到的第五个问题,但从来没有解释过“为什么不是静态的”。