javafx 8 兼容性问题 - FXML 静态字段
Posted
技术标签:
【中文标题】javafx 8 兼容性问题 - FXML 静态字段【英文标题】:javafx 8 compatibility issues - FXML static fields 【发布时间】:2021-12-08 17:11:18 【问题描述】:我设计了一个在 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)
我发现这是在控制器类的初始化方法中,我无法在任何静态组件中使用内置方法。 (例如: staticMyTextField.setText() 在 java 8 中导致问题,但在 java 7 中没有)。我无法在 javafx 指南中找到有关此内容的任何记录。有人可以提供一些关于为什么这会导致 Java 8 出现问题的想法吗?如果有的话,还可以分享与此相关的文档。
【问题讨论】:
能否提供代码示例? 请参考此链接。这在 JDK 7 中有效,但在 jdk 8 中无效 ***.com/questions/8434787/… 请帮忙? 【参考方案1】:听起来您正试图将TextField
注入静态字段。类似的东西
@FXML
private static TextField myTextField ;
这显然在 JavaFX 2.2 中有效。它在 JavaFX 8 中不起作用。由于没有官方文档支持这种用法,它并没有真正违反向后兼容性,但公平地说,关于 FXMLLoader
所做的事情的文档非常糟糕。
将@FXML
-injected 字段设为静态并没有多大意义。加载 FXML 文件时,它会为 FXML 文件中的每个元素创建新对象。一个新的控制器实例与对FXMLLoader.load(...)
的每次调用相关联,并且该控制器 instance 中的字段被注入为 FXML 元素创建的相应对象。因此注入的字段必须特定于控制器实例。如果您在控制器中有一个静态注入字段,并且您两次加载相同的 FXML 文件并在 UI 中显示两次,那么您将无法引用两组控件。
更新:在 cmets 中回答问题
特别是,不要使用静态字段来使它们可以从类外部访问。静态字段具有属于类的单个值,而不是类的每个实例的值,并且只有在有意义的情况下才应该做出使字段静态的决定。换句话说,static
定义 范围,而不是 可访问性。要允许访问实例数据,您只需要有对实例的引用。 FXMLLoader
有一个getController()
方法,允许您检索对控制器的引用。
相关的一点:从控制器公开 UI 控件也不是一个好主意。您应该改为公开数据。例如,不是在控制器中定义getTextField()
方法,而是定义一个textProperty()
方法,该方法返回一个代表TextField
内容的StringProperty
。这样做的原因是,当您的老板来到办公室并告诉您他希望将TextField
替换为TextArea
或ComboBox<String>
或其他一些控件时,这将变得更加困难如果控制器之外的类正在使用您的TextField
。与如何将数据呈现给用户的实现相比,控制器表示的数据结构发生变化的可能性要小得多。
举几个例子
Passing Parameters JavaFX FXML JavaFX Data Mangement Access fields from another Controller in 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【讨论】:
谢谢詹姆斯。无论如何可以从其他控制器类访问组件的内容?例如,我在一个控制器类的 listView 中显示了一些内容(复选框和标签)。如果我想从另一个控制器类访问它而不更改或更改其内容我应该怎么做? +1 以获得很好的解释。我一直认为静态在这些情况下是有意义的。你仍然可以使用静态直到8b109。这是迄今为止我在这方面看到的第 5 个问题,但从来没有解释过“为什么不是静态的”。以上是关于javafx 8 兼容性问题 - FXML 静态字段的主要内容,如果未能解决你的问题,请参考以下文章
JavaFX 8:javafx.fxml.LoadException - fxmlLoader.load() 不起作用,但 fxmlLoader.getController() 起作用
JavaFX 8 - 每个选项卡带有单独的 FXML 和控制器的选项卡和选项卡