浅析SpringBoot如何做定时任务:@EnableScheduling和@Scheduled的使用、参数配置以及需要注意的内存问题

  定时任务相当于闹钟,在什么时间做什么事情(执行什么命令/脚本)。

  @EnableScheduling 在配置类上使用,开启计划任务的支持  ——  用于类上。

  @Scheduled 来声明这是一个任务,包括cron、fixDelay、fixRate等类型  ——  用于方法上,需先开启计划任务的支持

一、如何使用

1、pom.xml 中导入必要的依赖

2、启动类里面使用@EnableScheduling 注解开启功能,自动扫描

@SpringBootApplication
@EnableScheduling //开启定时任务
public class MainApplication {
    ......
}

3、在类上加上:在任务方法上写 @Scheduled

@Service
public class TestService2 {
    private static final SimpleDateFormat format = new SimpleDateFormat("HH:mm:ss");
 
    //初始延迟1秒,每隔2秒
    @Scheduled(fixedRateString = "2000",initialDelay = 1000)
    public void testFixedRate(){
        System.out.println("fixedRateString,当前时间:" +format.format(new Date()));
    }
 
    //每次执行完延迟2秒
    @Scheduled(fixedDelayString= "2000")
    public void testFixedDelay(){
        System.out.println("fixedDelayString,当前时间:" +format.format(new Date()));
    }
 
    //每隔3秒执行一次
    @Scheduled(cron="0/3 * * * * ?")
    public void testCron(){
        System.out.println("cron,当前时间:" +format.format(new Date()));
    }
}

浅析SpringBoot如何做定时任务:@EnableScheduling和@Scheduled的使用、参数配置以及需要注意的内存问题

二、Scheduled注解中有以下几个参数:

1、cron表达式  ——  比如你要设置每天什么时候执行,就可以用它。

cron表达式,有专门的语法,而且感觉有点绕人,不过简单来说,大家记住一些常用的用法即可,特殊的语法可以单独去查。cron一共有7位,但是最后一位是年,可以留空,所以我们可以写6位:
第一位,表示秒,取值0-59
第二位,表示分,取值0-59
第三位,表示小时,取值0-23
第四位,日期天/日,取值1-31
第五位,日期月份,取值1-12
第六位,星期,取值1-7,星期一,星期二...,注:不是第1周,第二周的意思;另外:1表示星期天,2表示星期一。
第7为,年份,可以留空,取值1970-2099

  cron中,还有一些特殊的符号,含义如下:

(*) 星号:可以理解为每的意思,每秒,每分,每天,每月,每年...
(?) 问号:问号只能出现在日期和星期这两个位置。
(-) 减号:表达一个范围,如在小时字段中使用“10-12”,则表示从10到12点,即10,11,12
(,) 逗号:表达一个列表值,如在星期字段中使用“1,2,4”,则表示星期一,星期二,星期四
(/) 斜杠:如:x/y,x是开始值,y是步长,
比如在第一位(秒) 0/15就是,从0秒开始,每15秒,最后就是0,15,30,45,60 另:*/y,等同于0/y

  下面列举几个例子供大家来验证:

0 0 3 * * ?     每天3点执行
0 5 3 * * ?     每天3点5分执行
0 5 3 ? * *     每天3点5分执行,与上面作用相同
0 5/10 3 * * ?  每天3点的 5分,15分,25分,35分,45分,55分这几个时间点执行
0 10 3 ? * 1    每周星期天,3点10分 执行,注:1表示星期天    
0 10 3 ? * 1#3  每个月的第三个星期,星期天 执行,#号只能出现在星期的位置

2、zone  ——  表示执行时间的时区

3、fixedDelay 和 fixedDelayString  ——  表示一个固定延迟时间执行,上个任务完成后,延迟多长时间执行

4、fixedRate 和 fixedRateString  ——  表示一个固定频率执行,上个任务开始后,多长时间后开始执行

5、initialDelay 和initialDelayString表示一个初始延迟时间,第一次被调用前延迟的时间

  需要注意的是 cron、fixedDelay、fixedRate 必须得有一个,否则会报错:Encountered invalid @Scheduled method 'updateBookNum': Exactly one of the 'cron', 'fixedDelay(String)', or 'fixedRate(String)' attributes is required

三、注解参数区别

  @Scheduled注解可以控制方法定时执行,其中有三个参数可选择:

1、fixedDelay控制方法执行的间隔时间,是以上一次方法执行完开始算起,如上一次方法执行阻塞住了,那么直到上一次执行完,并间隔给定的时间后,执行下一次。

2、fixedRate是按照一定的速率执行,是从上一次方法执行开始的时间算起,如果上一次方法阻塞住了,下一次也是不会执行,但是在阻塞这段时间内累计应该执行的次数,当不再阻塞时,一下子把这些全部执行掉,而后再按照固定速率继续执行。

3、cron表达式可以定制化执行任务,但是执行的方式是与fixedDelay相近的,也是会按照上一次方法结束时间开始算起。

4、initialDelay 。如: @Scheduled(initialDelay = 10000,fixedRate = 15000),这个定时器就是在上一个的基础上加了一个initialDelay = 10000,意思就是在容器启动后,延迟10秒后再执行一次定时器,以后每15秒再执行一次该定时器。

四、Scheduled定时任务会导致内存泄漏吗?

1、要看你的定时任务是做什么的,只是调用无状态的方法是不会占用内存的,就像你用for循环执行一个任务很多次一样。但是,如果你在定时任务里进行一些影响全局的操作,例如不停地往一个List里append新的内容,当然会造成内存泄漏。或者说你在定时任务里新建一个以当前时间为文件名的文本文件,里面写上一些内容,一段时间后你的磁盘空间都会爆满的。总而言之会不会造成内存泄漏往往不是框架的问题,而是代码自己的问题。

2、尽管Java提供了自动的垃圾回收机制,但是这不代表说开发者就可以完全不管程序内对象和数据结构的使用限制了,比如最常见的场景,假设你把new完的对象放在队列或者Map里,然后这个定时任务不停地添加这种对象,那最后肯定会OOM的。GC只能回收那些没有被引用的对象,而如果你个个都引用并且并不释放(方法结束或者主动置为null,或者将存放的集合销毁),那再无敌的GC也帮不了你。所以不要把所有希望都寄托在GC上面。

  所以在使用定时器的过程中,需要注意定时器里的内存溢出问题。定时器本身不会导致内存溢出,就是怕定时器里的代码处理不好,加入无法被GC回收的对象,堆积导致内存溢出。
上一篇:SpringBoot使用Spring中自带的定时任务


下一篇:运行时动态修改@Scheduled注解的定时任务