问题描述:
- 使用 java.util.TimerTask 任务中断(遇到未抓取异常时后续任务不执行);
2. 使用 java.util.concurrent.Executors.xxx 线程池对于未抓取异常无异常输出,无法定位问题;
定位方法:
对比 TimerTask/Executors/Quartz 三者对抛出异常的反应
| TimerTask | Executors | Quartz |
是否中断后续任务 | 是 | 否 | 否 |
是否有异常输出 | 是(uncaught) | 无 | 是(log4j) |
问题解决:
建议:
- 对于定时任务,建议使用 quartz 库。(使用 TimerTask 有未抓取异常时会导致当前Timer终止,后续任务无法执行)
- 线程池有未抓取异常时不会导致后续任务中断,但是没有异常信息的输出,难以定位问题,对此,有三种方案:
- )自己封装一个RunnableWrapper、CallableWrapper,将传入的任务包裹其中,在run方法中try-catch,并统一对未抓取异常进行异常处理/输出;
- )继承ThreadPoolExecutor的方式来创建线程池,重写其afterExecute(Thread, Throwable)方法,判断第二个参数非空时获得其异常信息,统一进行处理;(注意:仅对 executor.execute(..) 启动的任务有效,executor.submit(..)的无效)
- )创建线程池时,传入自定义的ThreadFactory,在其创建线程时,使用setUncaughtExceptionHandler(..)设置每个线程的未抓取异常处理句柄,统一进行处理;(注意:仅对 executor.execute(..) 启动的任务有效,executor.submit(..)的无效)