Java线程池中任务异常默认静默丢失,因Runnable异常由未配置的UncaughtExceptionHandler忽略,Callable异常需显式调用Future.get()才能暴露;正确捕获方式包括任务内try-catch、自定义ThreadFactory设置异常处理器、重写afterExecute()钩子方法。
Java线程池中任务抛出的异常,**默认不会被主线程捕获,也不会打印堆栈,容易静默丢失**。这是开发者常踩的坑——看似任务执行了,实则因异常中断却毫无感知。
ThreadPoolExecutor 执行任务时,若 Runnable 任务中抛出未检查异常(如 RuntimeException),线程池会捕获它并调用 Thread.getUncaughtExceptionHandler() 处理;但默认的 handler 什么也不做。而 Callable 任务的异常会被包装进 ExecutionException,**必须显式调用 get() 才能暴露**。
new Runnable() { public void run() { try { doWork(); } catch (Exception e) { log.error("task failed", e); } } }
new ThreadFactoryBuilder().setUncaughtExceptionHandler((t, e) -> log.error("thread {} crashed", t.getName(), e)).build()
thrown != null 才是异常情况,且该方法在任务执行完后由工作线程调用
submit(Callable) 返回 Future,它的 get() 方法是异常曝光的唯一出口。不调用 get(),异常就永远藏在 Future 内部。
handle() 或 exceptionally() 捕获异常get(5, TimeUnit.SECONDS))避免无限阻塞如果任务频繁抛异常且未处理,可能引发连锁问题:
基本上就这些。关键是记住:线程池不替你处理异常,它只负责调度。异常在
