0

在我的 JavaFX 应用程序中,我想在出现意外异常时显示错误对话框并退出应用程序。所以在我的主要方法中,我在启动应用程序之前设置了一个默认的未捕获异常处理程序:

setDefaultUncaughtExceptionHandler((thread, cause) -> {
    try {
        cause.printStackTrace();
        final Runnable showDialog = () -> {
           // create dialog and show
        };
        if (Platform.isFxApplicationThread()) {
           showDialog.run();
        } else {
           runAndWait(showDialog);
        }
    } catch (Throwable t) {
        // ???
    } finally {
        System.exit(-1);
    }
});

launch(MyApp.class);

说明:当在 JavaFX 应用程序线程 (FXAT) 上执行未捕获的异常处理程序时,我只运行用于显示对话框的代码。当 FXAT 不调用异常处理程序时,这当然不起作用。在这种情况下,我必须将代码推送到 FXAT。但我不能使用Platform.runLater,因为我的应用程序会在显示对话框之前退出。因此,我制作了一个自定义方法,该方法runAndWait在内部通过 推送可运行对象Platform.runLater,但一直等到可运行对象的执行(使用一些倒计时锁存机制)。

现在的问题是:当我的 start() 方法中发生异常时,我的应用程序就会卡住。因为它试图等到执行显示的对话框,但 FXAT 从不执行此操作。我猜这是因为当 start() 方法因异常而失败时,FXAT 就死了?我不确定这是否是 start() 方法的特殊情况,或者这是否会在引发异常且未在 FXAT 执行的代码中捕获的任何情况下发生。

据我所知,在 Swing 中,EDT 是一个由多个线程组成的复杂架构。当 EDT 上的某些执行失败时,并不是整个 Swing 都崩溃了。但这里似乎发生了什么?

那么我可以在这里做什么?如何向用户显示应用程序无法启动?

4

1 回答 1

1

出色地....

我有一个解决方案,但我不特别推荐它。默认情况下,Application.launch()会捕获 start 方法抛出的异常,退出 FX 平台,然后重新抛出异常。由于在执行默认的未捕获异常处理程序时 FX 应用程序线程已关闭,因此等待 FX 应用程序线程上发生的事情只会无限期地阻塞。

FX 应用程序在 Web 启动中运行时例外。启动器对此进行检查的方式是检查安全管理器是否存在。所以一个(非常非常难看的)解决方法是安装一个安全管理器,这样它看起来就像你在 web 启动模式下运行一样。此行将安装一个许可安全管理器:

System.setSecurityManager(new SecurityManager(){
    @Override
    public void checkPermission(Permission perm) {}
});

这是一个SSCCE:

import java.lang.Thread.UncaughtExceptionHandler;
import java.security.Permission;
import java.util.concurrent.FutureTask;

import javafx.application.Application;
import javafx.application.Platform;
import javafx.scene.control.Alert;
import javafx.scene.control.Alert.AlertType;

public class ShowDialogOnException  {

    public static final UncaughtExceptionHandler ALERT_EXCEPTION_HANDLER = (thread, cause) -> {
        try {
            cause.printStackTrace();
            final Runnable showDialog = () -> {
               Alert alert = new Alert(AlertType.ERROR);
               alert.setContentText("An unknown error occurred");
               alert.showAndWait();
            };
            if (Platform.isFxApplicationThread()) {
               showDialog.run();
            } else {
               FutureTask<Void> showDialogTask = new FutureTask<Void>(showDialog, null);
               Platform.runLater(showDialogTask);
               showDialogTask.get();
            }
        } catch (Throwable t) {
            t.printStackTrace();
        } finally {
            System.exit(-1);
        }
    };



    public static void main(String[] args) {
        System.setSecurityManager(new SecurityManager(){
            @Override
            public void checkPermission(Permission perm) {}
        });
        Thread.setDefaultUncaughtExceptionHandler(ALERT_EXCEPTION_HANDLER);
        Application.launch(App.class, args);
    }
}

和一个测试应用程序:

import javafx.application.Application;
import javafx.stage.Stage;

public class App extends Application {

    @Override
    public void start(Stage primaryStage) throws Exception {
        throw new Exception("An exception");
    }

    @Override
    public void stop() {
        System.out.println("Stop");
    }

}

正如我所说,这确实是一个大黑客,除非您别无选择,否则我不建议您这样做。

于 2016-03-12T16:21:40.130 回答