12

我试过了

Thread.setDefaultUncaughtExceptionHandler...


在 main 和 start(Stage primaryStage) 方法中。它不工作。
我也试过

public static void main(String[] args) {
 try {
  launch(args);
 }catch(Throwable t) {
  System.out.println(t.getMessage);
 }
}


异常堆栈跟踪。

在 javafx.concurrent.Task$TaskCallable$2.run(Task.java:1251) 在 com.sun.javafx.application.PlatformImpl$3.run(PlatformImpl.java:141) 在 com.sun.glass.ui.gtk.GtkApplication ._runLoop(Native Method) at com.sun.glass.ui.gtk.GtkApplication$1$1.run(GtkApplication.java:56) at java.lang.Thread.run(Thread.java:662)

感谢您的帮助。

4

3 回答 3

17

如果您检查代码Platform.runLater()(见下文),您将看到异常被吞没(第 146 / 147 行),因此默认的未捕获异常处理程序将无法捕获它们 - 基于那段代码,我不'认为除了在可运行文件中包含 try/catch 块外,您别无选择。

请注意,此问题已报告(需要登录 - 注册是免费的)并且应该在 Lombard 中修复(= Java FX 8.0 将于明年与 Java 8 一起发布)。

您也可以创建一个实用方法并调用

Platform.runLater(getFxWrapper(yourRunnable));

public static Runnable getFxWrapper(final Runnable r) {
    return new Runnable() {

        @Override
        public void run() {
            try {
                r.run();
            } catch (Exception e) {
                //here you probably want to log something
                System.out.println("Found an exception");
            }
        }
    };
}

代码Platform.runLater

  120     private static void runLater(final Runnable r, boolean exiting) {
  121         if (!initialized.get()) {
  122             throw new IllegalStateException("Toolkit not initialized");
  123         }
  124 
  125         pendingRunnables.incrementAndGet();
  126         waitForStart();
  127 
  128         if (SystemProperties.isDebug()) {
  129             Toolkit.getToolkit().pauseCurrentThread();
  130         }
  131 
  132         synchronized (runLaterLock) {
  133             if (!exiting && toolkitExit.get()) {
  134                 // Don't schedule a runnable after we have exited the toolkit
  135                 pendingRunnables.decrementAndGet();
  136                 return;
  137             }
  138 
  139             Toolkit.getToolkit().defer(new Runnable() {
  140                 @Override public void run() {
  141                     try {
  142                         r.run();
  143                         pendingRunnables.decrementAndGet();
  144                         checkIdle();
  145                     } catch (Throwable t) {
  146                         System.err.println("Exception in runnable");
  147                         t.printStackTrace();
  148                     }
  149                 }
  150             });
  151         }
  152     }
于 2012-10-09T10:46:44.210 回答
2

一些托管线程(例如 UI 事件处理程序和 ExecutorServices)自己捕获 Throwable 以避免线程死亡。只有死掉的线程才会使用这个 UncaughtExceptionHandler。如果要捕获抛出的异常,则必须在可以抛出这些异常的方法中执行此操作。

如果 UI 事件处理程序有报告异常的方法,那将是另一种方法。

您的第二个示例将捕获该线程中引发的异常。其他线程上抛出的异常将被该线程捕获。

于 2012-09-07T13:24:10.557 回答
2

将 EventDispatcher 设置为根节点对我有用。

 public class Frame extends Pane {
    Frame() {
        setEventDispatcher(new EventDispatcher() {

            @Override
            public Event dispatchEvent(Event event, EventDispatchChain chain) {
                try {
                    return chain.dispatchEvent(event);
                } catch (final Exception e) {
                    // handle all the exceptions here 
                    return null;
                }
            }
        });
    }
}
于 2014-06-27T12:44:57.597 回答