JavaFX 2 - Захват всех исключений во время выполнения

Я попробовал

Thread.setDefaultUncaughtExceptionHandler...


в основном, а также в начале (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)   at com.sun.glass.ui.gtk.GtkApplication._runLoop (собственный метод) в com.sun.glass.ui.gtk.GtkApplication $1 $1.run(GtkApplication.java:56)   в java.lang.Thread.run(Thread.java:662)

Спасибо за помощь.

Ответ 1

Если вы проверите код для Platform.runLater() (см. ниже), вы увидите, что исключения проглатываются (строки 146/147), поэтому обработчик исключенных исключений по умолчанию не сможет их поймать - на основе этого кусок кода, я не думаю, что у вас есть какие-то варианты, кроме включения блоков try/catch в ваши runnables.

Обратите внимание, что эта проблема была отправлена ​​ (требуется регистрация в системе - бесплатно) и должна быть исправлена ​​в 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     }

Ответ 2

Некоторые управляемые потоки, такие как обработчик событий пользовательского интерфейса и ExecutorServices, захватывают себя, чтобы избежать смерти нитей. Только потоки, которые умирают, будут использовать этот UncaughtExceptionHandler. Если вы хотите захватить исключение, вы должны сделать thsi в методе, который может вызывать эти исключения.

Если обработчик событий пользовательского интерфейса имеет способ сообщения об исключениях, это будет другой метод.

В следующем примере вы увидите исключение, созданное в этом потоке. Исключение, брошенное на другие потоки, будет зависеть от этого потока.

Ответ 3

Настройка EventDispatcher для корневого Node работала для меня.

 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;
                }
            }
        });
    }
}