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



我设计了一个javafx应用程序,它在jdk 7中运行良好。当我尝试在java8中运行它时,我得到了以下异常:

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中引起问题的想法吗?并共享与此相关的文档(如果有的话)。

听起来您正试图将TextField注入静态字段。类似的东西

@FXML
private static TextField myTextField ;

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

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


从FXML引用的@FXML标记的静态方法也将不起作用。

例如,使用:

@FXML
private static void changeUser(ActionEvent e) {
    // System.out.println("test");
}

和:

<Button text="Change User" onAction="#changeUser"/>

这将失败:

javafx.fxml.LoadException: Error resolving onAction='#changeUser', either the event handler is not in the Namespace or there is an error in the script.

要解决此问题,只需从方法指定中删除静态修饰符即可。


更新:回复评论中的问题

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

相关的一点是:从控制器中公开UI控件也不是一个好主意。您应该公开数据。例如,不是在控制器中定义getTextField()方法,而是定义返回表示TextField的内容的StringPropertytextProperty()方法。原因是,当你的老板来到办公室,告诉你他想用TextAreaComboBox<String>或其他控件替换TextField时,如果控制器之外的类使用你的TextField,那就更难了。与如何向用户显示数据的实现方式相比,控制器所表示的数据结构变化的可能性要小得多。

例如

  • MVC在JavaFx中的应用
  • 传递参数JavaFX FXML
  • JavaFX数据管理
  • 从JavaFX中的另一个控制器访问字段
  • https://github.com/james-d/LeastCommonMultipleCalculator
  • https://github.com/james-d/Nested-Controller-Example
  • https://github.com/james-d/Dialog-FXML-Example
  • https://github.com/james-d/Shared-Data-Controller

最新更新