Ответ 1
Я не уверен, что это то, что вы ищете, но есть обработчик исключений, которые завершают потоки. Это обработчик для любого исключения, которое явно не попадает в цель потока.
По умолчанию "обработчик исключенных исключений" просто вызывает printStackTrace()
в Throwable
для печати трассировки стека на System.err
. Однако вы можете заменить своим собственным UncaughtExceptionHandler
, который вместо этого записывает исключение в log4j:
class Log4jBackstop implements Thread.UncaughtExceptionHandler {
private static Logger log = Logger.getLogger(Log4jBackstop.class);
public void uncaughtException(Thread t, Throwable ex) {
log.error("Uncaught exception in thread: " + t.getName(), ex);
}
}
Если вы используете фреймворк исполнителя, которому вы передаете объект Runnable
, его потоки, вероятно, имеют свой собственный блок catch
, который предотвращает доступ исключений к обработчику неперехваченных исключений. Если вы хотите поймать исключения Runtime
там, один способ сделать это - обернуть каждую задачу в обертки журнала, например:
class Log4jWrapper {
private final Logger log;
private final Runnable target;
Log4jWrapper(Logger log, Runnable target) {
this.log = Objects.requireNonNull(log);
this.target = Objects.requireNonNull(target);
}
public void run() {
try {
target.run();
} catch(RuntimeException ex) {
log.error("Uncaught exception.", ex);
throw ex;
}
}
}
...
Runnable realTask = ...;
executor.submit(new Log4jWrapper(Logger.getLogger(Whatever.class), realTask));