首页 > 解决方案 > 调用 Stage.show() 引发的无法捕获的异常

问题描述

编辑:我很幸运,James_D 在下面的评论中迅速发现并解释了这种现象。现在找到了使用PropertyValueFactory(对于 Groovy 用户)的替代方法,我想我会添加自己的答案,以帮助任何不幸的人在未来对此感到困惑。


规格 Java:11。操作系统:Linux Mint 18.3。所有这些应用程序和测试代码都是用 Groovy (2.5.10) 编写的,但这不应该特别相关(即同样的现象应该发生在用 Java 编写的等效项目中)。

App.start()我有一个从where Appextends调用的方法javafx.application.Application(所以它在 J​​AT/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 列“单元格值工厂”,它规定类实例的字段名称将被设置为TreeItems 的值。

值类被称为Task... 并且它的第一个字段是String title. 我为此添加了第一家工厂。但是(因为我是通过 TDD 进行的)我已经为第二列属性添加了工厂dueDate,但是我还没有在Task. 自然地,确保已设置具有正确属性名称的工厂的测试失败了。但我更感兴趣的是installDist此时我执行 Gradle(即分发自包含的可执行文件)时会发生什么。

这运行:显示窗口。我可以看到TreeItems 被赋予了正确的标题。在终端中,我看到由于我未能在实例中包含必填字段 ( 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( IllegalStateis a RuntimeException) 并没有被上面的内容抓住try ... catch。为什么不?有什么办法可以捕捉到这样的异常吗?

这显然是 JAT:您可以App.start()在该堆栈跟踪中看到(以及GraphBuilder.attachAndShow()

我刚刚进行了几个实验:在这个方法中先抛出一个 a RuntimeException,然后是一个标准,在方法中的 4 行中的任何一行之前ExceptionattachAndShow()

在这两种情况下,(分布式)应用程序都无法运行,并且什么也没有显示......但是日志消息确实被记录了,即这些都被捕获了。完全符合预期的行为,因此 JAT 本身似乎不是“罪魁祸首”。

我还注意到上面显示的跟踪不是“正常”的堆栈跟踪:前两行表明有东西正在处理它,而且它似乎很不幸地吞下了它。至少我希望能够记录这些问题!

标签: javamultithreadingexceptionjavafxgroovy

解决方案


基本答案在这里

不过,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(请参见此处)以及对显示其实例的类的一些剪裁。


推荐阅读