在Java多线程中,子线程异常不会自动传递给主线程,需通过特定方式捕获。2. 可在run方法中使用try-catch直接处理异常。3. 通过setUncaughtExceptionHandler设置线程未捕获异常的处理器。4. 使用Callable和Future时,future.get()会抛出ExecutionException,通过getCause()获取原始异常。5. 自定义ThreadFactory可为线程池中的线程统一设置异常处理器。6. 根据场景选择方法:简单任务用try-catch,线程池推荐结合Callable与UncaughtExceptionHandler,确保异常被有效处理。

在Java多线程中,捕获异常比单线程复杂,因为子线程中的异常不会自动传递给主线程。如果不在子线程中显式处理,异常可能被忽略,导致程序出错却无提示。要正确捕获多线程中的异常,有几种有效方式。
使用try-catch在run方法中捕获
最直接的方式是在Thread的run()方法或Runnable实现中使用try-catch块。
例如:
new Thread(() -> {
try {
// 可能抛出异常的代码
int result = 10 / 0;
} catch (Exception e) {
System.out.println("子线程捕获异常: " + e.getMessage());
}
}).start();
这种方式简单有效,适用于简单的任务处理。
立即学习“Java免费学习笔记(深入)”;
通过UncaughtExceptionHandler处理未捕获异常
每个线程都可以设置一个UncaughtExceptionHandler,用于处理未被捕获的异常。
设置方式:
Thread thread = new Thread(() -> {
throw new RuntimeException("测试异常");
});
thread.setUncaughtExceptionHandler((t, e) -> {
System.out.println("线程 " + t.getName() + " 发生异常: " + e.getMessage());
});
thread.start();
也可以为所有线程设置默认处理器:
Thread.setDefaultUncaughtExceptionHandler((t, e) -> {
System.out.println("全局异常处理器: 线程 " + t.getName() + ", 异常: " + e.getMessage());
});
使用Callable和Future获取异常
当使用ExecutorService时,推荐使用Callable代替Runnable,因为它可以返回结果并抛出异常。
Future.get()方法会将执行中的异常封装为ExecutionException抛出。
ExecutorService executor = Executors.newSingleThreadExecutor(); Futurefuture = executor.submit(() -> { throw new RuntimeException("任务执行失败"); }); try { Integer result = future.get(); // 此处会抛出ExecutionException } catch (ExecutionException e) { System.out.println("任务异常: " + e.getCause().getMessage()); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } executor.shutdown();
通过e.getCause()可以获得原始异常对象。
自定义线程工厂统一设置异常处理器
在使用线程池时,可以通过自定义ThreadFactory为每个线程设置统一的异常处理逻辑。
ThreadFactory factory = r -> {
Thread t = new Thread(r);
t.setUncaughtExceptionHandler((thread, e) ->
System.err.println("工厂创建的线程异常: " + e.getMessage())
);
return t;
};
ExecutorService executor = Executors.newFixedThreadPool(2, factory);
基本上就这些。根据使用场景选择合适的方法:简单任务用try-catch,线程池推荐结合Callable和UncaughtExceptionHandler,确保异常不被遗漏。










