上一节,我们通过自己启动一个线程来执行定时任务,其实我们还有许多方便执行定时任务的方法。quartz是非常出名的一个定时任务框架,spring也提供了一个定时任务框架,两者使用方法基本一样,但是spring的定时任务框架其实是quartz的简化版,他不支持任务的持久化,不支持分布式。这里,我们简单介绍怎么启用spring的定时任务框架。
1、开启注解
使用@EnableScheduling来开启注解。
2、Fixed
我们注释掉之前CmsController里面的”定时任务“
新建一个CmsTimer类
使用@Scheduled注解方法,fixedRate表示每隔多少毫秒执行一次。我们还可以通过设置initialDelay来控制第一次执行的时间,如没设置initialDelay的时候,初始化完就会马上执行一次,如果我们想1分钟后再执行第一次任务,就可以设置initialDelay=60*1000。fixedDelay与fixedRate的区别是,前者是在任务执行完后延迟多久,后者是在任务开始后延迟多久。
3、cron表达式
@Scheduled(cron = "0 * * * * ?")
上面的cron表达式,表示每分钟执行一次,第一次执行是秒钟置0的时候。cron表达式具体的用法大家自行查找。
4、多任务
如果我们代码中定义了多个定时任务,恰好几个任务会同时执行,会发现任务会有延迟或丢失的可能,这是因为,spring提供的定时任务其实底层就是ScheduledThreadPoolExecutor,默认只有一个任务线程,一个线程执行完当前任务,才能执行另外一个任务。如果碰到任务执行不过来的情况,我们可以加大线程池,通过spring.task.scheduling.pool.size配置。
我们还可以配置异步线程,spring将默认开启8个线程来执行异步操作。
5、总结
这节我们讲了怎么在spring里面快速实现定时任务,文章开头我们说了,这种方式不支持任务持久化,不支持分布式,如果需要这两个特性,我们可以使用quartz,或者自己在spring定时任务上做一点点开发工作。例如,如果要支持集群部署,有时我们可以在配置文件里面增加一个变量,配置是否开启定时任务,这样就会只有一台机器跑定时任务,这也是我们常见的做法。如果要真正的分布式,我们也可以通过redis加锁,或者zookeeper的方式实现。
代码:
https://github.com/www15119258/springboot-study/tree/branch35