java - 调用 Stage.show() 引发的无法捕获的异常
问题描述
编辑:我很幸运,James_D 在下面的评论中迅速发现并解释了这种现象。现在找到了使用PropertyValueFactory
(对于 Groovy 用户)的替代方法,我想我会添加自己的答案,以帮助任何不幸的人在未来对此感到困惑。
规格 Java:11。操作系统:Linux Mint 18.3。所有这些应用程序和测试代码都是用 Groovy (2.5.10) 编写的,但这不应该特别相关(即同样的现象应该发生在用 Java 编写的等效项目中)。
App.start()
我有一个从where App
extends调用的方法javafx.application.Application
(所以它在 JAT/JavaFX 应用程序线程中运行):
class GraphBuilder {
...
def attachAndShow(Stage stage, Node rootNode) {
try {
App.instance.fxmlController.treeTableView.root = rootTreeItem
App.instance.fxmlController.treeTableView.showRoot = false
stage.scene = new Scene(rootNode, 1200, 800)
stage.show()
}catch( Throwable t ){
log.error( "got throwable in attachAndShow... $t.message", t )
}
}
我在一个“尖峰”模型(我刚刚跑到前面的未经测试的代码)模型上进行 TDD,在这个模型中TreeTableView
填充了我。我强调没有任何迹象表明尖峰有任何问题。
这样做的本质是设置TreeTableView
's 列“单元格值工厂”,它规定类实例的字段名称将被设置为TreeItem
s 的值。
值类被称为Task
... 并且它的第一个字段是String
title
. 我为此添加了第一家工厂。但是(因为我是通过 TDD 进行的)我已经为第二列属性添加了工厂dueDate
,但是我还没有在Task
. 自然地,确保已设置具有正确属性名称的工厂的测试失败了。但我更感兴趣的是installDist
此时我执行 Gradle(即分发自包含的可执行文件)时会发生什么。
这运行:显示窗口。我可以看到TreeItem
s 被赋予了正确的标题。在终端中,我看到由于我未能在实例中包含必填字段 ( dueDate
)导致的堆栈跟踪Task
:
Mar 23, 2020 6:16:25 PM javafx.scene.control.cell.TreeItemPropertyValueFactory getCellDataReflectively
WARNING: Can not retrieve property 'dueDate' in TreeItemPropertyValueFactory: javafx.scene.control.cell.TreeItemPropertyValueFactory@2d1e69d0 with provided class type: class core.Task
java.lang.IllegalStateException: Cannot get property dueDate
at com.sun.javafx.property.PropertyReference.getProperty(PropertyReference.java:194)
at javafx.scene.control.cell.TreeItemPropertyValueFactory.getCellDataReflectively(TreeItemPropertyValueFactory.java:185)
at javafx.scene.control.cell.TreeItemPropertyValueFactory.call(TreeItemPropertyValueFactory.java:158)
at javafx.scene.control.cell.TreeItemPropertyValueFactory.call(TreeItemPropertyValueFactory.java:136)
at javafx.scene.control.TreeTableColumn.getCellObservableValue(TreeTableColumn.java:576)
at javafx.scene.control.TreeTableColumn.getCellObservableValue(TreeTableColumn.java:561)
at javafx.scene.control.TreeTableCell.updateItem(TreeTableCell.java:632)
at javafx.scene.control.TreeTableCell.indexChanged(TreeTableCell.java:457)
at javafx.scene.control.IndexedCell.updateIndex(IndexedCell.java:120)
at javafx.scene.control.skin.TableRowSkinBase.updateCells(TableRowSkinBase.java:539)
at javafx.scene.control.skin.TreeTableRowSkin.updateCells(TreeTableRowSkin.java:276)
at javafx.scene.control.skin.TableRowSkinBase.<init>(TableRowSkinBase.java:159)
at javafx.scene.control.skin.TreeTableRowSkin.<init>(TreeTableRowSkin.java:102)
at javafx.scene.control.TreeTableRow.createDefaultSkin(TreeTableRow.java:529)
at javafx.scene.control.Control.doProcessCSS(Control.java:897)
at javafx.scene.control.Control.access$000(Control.java:83)
at javafx.scene.control.Control$1.doProcessCSS(Control.java:89)
at com.sun.javafx.scene.control.ControlHelper.processCSSImpl(ControlHelper.java:67)
at com.sun.javafx.scene.NodeHelper.processCSS(NodeHelper.java:145)
at javafx.scene.Node.processCSS(Node.java:9529)
at javafx.scene.Node.applyCss(Node.java:9616)
at javafx.scene.control.skin.VirtualFlow.setCellIndex(VirtualFlow.java:1715)
at javafx.scene.control.skin.VirtualFlow.getCell(VirtualFlow.java:1692)
at javafx.scene.control.skin.VirtualFlow.getCellLength(VirtualFlow.java:1801)
at javafx.scene.control.skin.VirtualFlow.computeViewportOffset(VirtualFlow.java:2639)
at javafx.scene.control.skin.VirtualFlow.layoutChildren(VirtualFlow.java:1245)
at javafx.scene.Parent.layout(Parent.java:1204)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Parent.layout(Parent.java:1211)
at javafx.scene.Scene.doLayoutPass(Scene.java:576)
at javafx.scene.Scene.preferredSize(Scene.java:1748)
at javafx.scene.Scene$2.preferredSize(Scene.java:393)
at com.sun.javafx.scene.SceneHelper.preferredSize(SceneHelper.java:66)
at javafx.stage.Window$12.invalidated(Window.java:1086)
at javafx.beans.property.BooleanPropertyBase.markInvalid(BooleanPropertyBase.java:110)
at javafx.beans.property.BooleanPropertyBase.set(BooleanPropertyBase.java:145)
at javafx.stage.Window.setShowing(Window.java:1174)
at javafx.stage.Window.show(Window.java:1189)
at javafx.stage.Stage.show(Stage.java:273)
at javafx.stage.Stage$show.call(Unknown Source)
at org.codehaus.groovy.runtime.callsite.CallSiteArray.defaultCall(CallSiteArray.java:47)
at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:115)
at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:119)
at core.GraphBuilder.attachAndShow(main.groovy:111)
at core.GraphBuilder$attachAndShow$4.call(Unknown Source)
at org.codehaus.groovy.runtime.callsite.CallSiteArray.defaultCall(CallSiteArray.java:47)
at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:115)
at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:135)
at core.App.start(main.groovy:56)
at com.sun.javafx.application.LauncherImpl.lambda$launchApplication1$9(LauncherImpl.java:846)
at com.sun.javafx.application.PlatformImpl.lambda$runAndWait$12(PlatformImpl.java:455)
at com.sun.javafx.application.PlatformImpl.lambda$runLater$10(PlatformImpl.java:428)
at java.base/java.security.AccessController.doPrivileged(Native Method)
at com.sun.javafx.application.PlatformImpl.lambda$runLater$11(PlatformImpl.java:427)
at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:96)
at com.sun.glass.ui.gtk.GtkApplication._runLoop(Native Method)
at com.sun.glass.ui.gtk.GtkApplication.lambda$runLoop$11(GtkApplication.java:277)
at java.base/java.lang.Thread.run(Thread.java:834)
this Throwable
( IllegalState
is a RuntimeException
) 并没有被上面的内容抓住try ... catch
。为什么不?有什么办法可以捕捉到这样的异常吗?
这显然是 JAT:您可以App.start()
在该堆栈跟踪中看到(以及GraphBuilder.attachAndShow()
)
我刚刚进行了几个实验:在这个方法中先抛出一个 a RuntimeException
,然后是一个标准,在方法中的 4 行中的任何一行之前。Exception
attachAndShow()
在这两种情况下,(分布式)应用程序都无法运行,并且什么也没有显示......但是日志消息确实被记录了,即这些都被捕获了。完全符合预期的行为,因此 JAT 本身似乎不是“罪魁祸首”。
我还注意到上面显示的跟踪不是“正常”的堆栈跟踪:前两行表明有东西正在处理它,而且它似乎很不幸地吞下了它。至少我希望能够记录这些问题!
解决方案
基本答案在这里。
不过,Groovy 用户可以沾沾自喜地切入一大片尖括号:我的解决方案变成了:
fxmlController.treeTableView.columns.get( 0 ).cellValueFactory =
new Callback() {
@Override
Binding call(Object cellDataFeatures ) {
// NB it turns out that the first "value" is a TreeItem
Task task = cellDataFeatures.value.value
Bindings.createStringBinding({ task.title })
}
}
或者粘贴一些最小的可能:
fxmlController.treeTableView.columns.get( 0 ).cellValueFactory =
new Callback<TreeTableColumn.CellDataFeatures, ObservableValue>() {
@Override
Binding call(TreeTableColumn.CellDataFeatures cellDataFeatures ) {
Task task = cellDataFeatures.value.value
Bindings.createStringBinding({ task.title })
}
}
对于 Groovy 的“Java lambda”替代品,我们使用闭包也就不足为奇了:
fxmlController.treeTableView.columns.get( 0 ).cellValueFactory
= { cellDataFeatures -> cellDataFeatures.value.titleProperty() }
请注意,这涉及到对 JavaFX 的一些了解Property
(请参见此处)以及对显示其实例的类的一些剪裁。
推荐阅读
- html - 如何创建指向框的自指向箭头
- regex - 正则表达式在第一个管道后获取所有字符串
- php - 使用 laravel 刀片中的 mpdf 包生成的 PDF 上不会显示 FontAwesome 5 图标?
- android - 如何使用设备令牌发送下行消息
- android - 在颤振中无法从 Android 30 中的模拟器打开相机
- node.js - 我对 bot 有不和谐的意图,但意图显示错误?
- filebeat - filebeat配置中的容器和docker输入有什么区别?
- mysql - 如何在mysql中将列的数据类型从文本更改为日期时间?
- c# - 变量应该如何传递到方法之外
- authentication - ERPNext:如何在没有密码的情况下生成用户会话?