前面出了一个20000+元的 Bug,现在来认真的看看 Spring Schedule 是如何处理定时任务的。

Spring Schedule 配置时间的三种方式

  • fixRate: 间隔时间,会根据首次执行时间预先安排固定的时间,错过立即执行
  • fixDelay: 间隔时间,上次结束时间后多少间隔后执行
  • cron: 指定具体时间,错过立即执行

fixRate 运行方式

fixRate 是会根据开始时间预先安排一系列的固定时间点,如果错过了安排的时间点,则立即执行,否则等待下一个执行时间点,如下图 fixRate=5000,每5秒执行一次,第二次执行超过了预先安排的点,导致第三次错过,则第三次立即执行,二第四次则在正常安排的时间点执行,值得注意的是第三次与第四次相隔仅一秒。

Spring Schedule fixRate图例

fixDelay 运行方式

fixDelay 就相对简单一些,上次执行完开始计时,到时间就执行。会有一种特殊情况,如果间隔时间到了,没有可用于执行的线程,那么下次获取到线程后立即执行。如下图 fixDelay=5000,每5秒执行一次:

Spring Schedule fixDelay图例

cron 运行方式

cron 可以指定具体的时间,和 fixRate 有些类似,错过立即执行。我前面的bug就是因为这个原因引起的,一台实例开始执行并加了锁,按理说其他实例相同时间则获取锁失败跳过执行,但另一个实例由于定时任务未能获取到执行线程,延后了,获取到线程后立即执行,导致这个时间不正常,而且前面的锁的时间不够长,导致重复执行。

TODO

一些可以扩展了解的地方

  1. schedule 自定义线程池
  2. 多实例环境下如何保证唯一
  3. 一些定时任务可选方案