JavaFX 8 (Lombard)全局异常处理

5
我目前正在尝试使用JavaFX 8构建一个应用程序,但我无法使未捕获的异常处理正常工作。根据这篇文章(https://bugs.openjdk.java.net/browse/JDK-8100937),它应该已经在JavaFX 8(Lombard)中得到修复/实现,但我在网络上找不到任何信息...
我不想采取hackish的方法,你能给我一些提示去哪里寻找更多信息吗?

你能给一个例子(代码)说明你正在做的导致了这个问题吗? - SnakeDoc
没有特定的代码会引起问题,我只是没有找到进一步了解新实现的异常处理的重点。全局异常处理旨在捕获未处理的异常,向用户打印一个简短的抱歉消息,然后退出。 - bash0r
2个回答

13

据我所了解,这并不复杂;您只需使用java.lang.Thread中的常规未捕获异常处理即可。

import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.layout.StackPane;
import javafx.stage.Stage;

public class UncaughtExceptionTest extends Application {

    @Override
    public void start(Stage primaryStage) {

        // start is called on the FX Application Thread, 
        // so Thread.currentThread() is the FX application thread:
        Thread.currentThread().setUncaughtExceptionHandler((thread, throwable) -> {
            System.out.println("Handler caught exception: "+throwable.getMessage());
        });

        StackPane root = new StackPane();
        Button button = new Button("Throw exception");
        button.setOnAction(event -> {
            throw new RuntimeException("Boom!") ;
        });
        root.getChildren().add(button);
        Scene scene = new Scene(root, 150, 60);
        primaryStage.setScene(scene);
        primaryStage.show();
    }

    public static void main(String[] args) {
        launch(args);
    }
}

这里难道不需要担心更多的线程吗?例如JavaFX应用程序线程?我认为你需要使用Platform.runLater()再做一次。 - User
好的,没问题,这个已经处理了 - 那其他线程呢?调用Thread.setUncaughtExceptionHandler(静态方法)是否可以捕获所有异常?Javadoc似乎没有明确说明。 - User
文档 对我来说看起来很清楚。如果线程本身没有设置未捕获异常处理程序,并且如果它所属的线程组没有覆盖uncaughtException方法,那么将调用在Thread.setDefaultUncaughtExceptionHandler()中设置的处理程序(如果有)。 - James_D
1
如果他们明确使用“所有”或“任何”线程的短语,而不是让读者通过静态上下文来推断,那么这将会更清晰一些。或者可能只是看起来太容易了,我下意识地持怀疑态度... - User

0

James_D的回答对我帮助很大,但是我忽略了他的评论。所以为了其他人遇到同样的问题,我把这个作为一个补充回答发出来。

在我的情况下,我的JavaFX应用程序必须面对和捕获内存不足错误。对于相同的代码,有时候OOM出现在JavaFX主线程上,有时候不是(可能是“InvokeLaterDispatcher”线程,“Timer”线程,“进程收割者”线程等),这取决于哪个线程首先耗尽了内存。因此,错误并不总是被捕获,应用程序也没有按照要求退出。

因此,为了确保我的应用程序记录并退出,我必须使用Thread.setDefaultUncaughtExceptionHandler(),正如James_D在他的评论中提到的那样:

Thread.setDefaultUncaughtExceptionHandler((thread, throwable) -> {
    System.out.println(throwable.getClass() + " detected from default UEH.\nWill exit now");
    System.exit(1);            
});

现在无论何时出现错误,都会被捕获。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接