使用TimerTask可以方便的实现定时任务的功能,但是如果使用不当,反而会带来隐患。

在使用TimerTask时,TimerTask中的代码必须要做异常处理,否则产生异常的时候,就挂掉了。
特别像使用MQ发送数据的时候,不会显式的要求你捕获异常,如果你忘记了,那么在某个时刻MQ异常的时候(比如网络异常),在发送数据到MQ失败的时候,TimerTask就挂掉了。

比如如下代码:

1
2
3
4
5
6
7
8
9
10
11
12
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext_*.xml");
final JmsSender jmsSender = ac.getBean(JmsSender.class);
Timer timer = new Timer();
// 1.TimerTask中不处理异常
timer.schedule(new TimerTask() {
    @Override
    public void run() {
        System.out.println("开始发送数据");
        jmsSender.sendTopicMsg("test.topic","hello,world");
        System.out.println("数据发送成功");
    }
},10000,5000);

这段代码没有做异常处理,我们看下执行结果:


最开始启动程序时,让MQ正常启动起来,这个时候TimerTask是正常工作的;在某个时刻关闭MQ,这个时候发现TimerTask中已经没有打印任何东西了,包括后面MQ恢复了也没有再打印,说明TimerTask已经挂掉了。
所以在使用TimerTask的时候要尤其注意这点,搞不好就踩着坑了。

处理方法有这么几种:
1.仍然使用TimerTask,但做异常处理;
2.使用单一线程的线程池来做;
3.使用线程,同样做异常处理。