SpringBoot之定时器

1、Timer 类

构造器

	//默认定时器名称
    public Timer() {
        this("Timer-" + serialNumber());
    }

	//以守护进程的方式运行此定时器
    public Timer(boolean isDaemon) {
        this("Timer-" + serialNumber(), isDaemon);
    }

	//自定义定时器名称
    public Timer(String name) {
        thread.setName(name);
        thread.start();
    }

    public Timer(String name, boolean isDaemon) {
        thread.setName(name);
        thread.setDaemon(isDaemon);
        thread.start();
    }

schedule

	//计划在指定的延迟后执行指定的任务,以毫秒为单位
    public void schedule(TimerTask task, long delay) {
        if (delay < 0)
            throw new IllegalArgumentException("Negative delay.");
        sched(task, System.currentTimeMillis()+delay, 0);
    }

	//计划在指定时间执行指定任务。如果时间已过,则计划立即执行任务
	// long delay 跟 Date time 的区别:delay 必须大于0,即未来时间;time可以是过去时间
    public void schedule(TimerTask task, Date time) {
        sched(task, time.getTime(), 0);
    }
    
    //将指定的任务安排为重复的固定延迟执行,从指定的延迟之后开始。
    //随后大约每隔一定的时间间隔重复执行。都是以毫秒为单位
	public void schedule(TimerTask task, long delay, long period) {
        if (delay < 0)
            throw new IllegalArgumentException("Negative delay.");
        if (period <= 0)
            throw new IllegalArgumentException("Non-positive period.");
        sched(task, System.currentTimeMillis()+delay, -period);
    }
    
    public void schedule(TimerTask task, Date firstTime, long period) {
        if (period <= 0)
            throw new IllegalArgumentException("Non-positive period.");
        sched(task, firstTime.getTime(), -period);
    }

	//以毫秒为单位,在指定的时间和指定的时间段安排指定的计时器任务执行。
	//如果周期为正,则计划重复执行任务;如果周期为零,则计划一次性执行任务
    private void sched(TimerTask task, long time, long period) {
        if (time < 0)
            throw new IllegalArgumentException("Illegal execution time.");

        // Constrain value of period sufficiently to prevent numeric
        // overflow while still being effectively infinitely large.
        if (Math.abs(period) > (Long.MAX_VALUE >> 1))
            period >>= 1;

        synchronized(queue) {
            if (!thread.newTasksMayBeScheduled)
                throw new IllegalStateException("Timer already cancelled.");

            synchronized(task.lock) {
                if (task.state != TimerTask.VIRGIN)
                    throw new IllegalStateException(
                        "Task already scheduled or cancelled");
                task.nextExecutionTime = time;
                task.period = period;
                task.state = TimerTask.SCHEDULED;
            }

            queue.add(task);
            if (queue.getMin() == task)
                queue.notify();
        }
    }
    public static void main(String[] args) throws InterruptedException {
        demo01();
    }

    public static void demo01() throws InterruptedException {
        Timer timer = new Timer();
        log.info("=============第一次启动定时器=============");
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                log.info("一次性执行,启动定时器延迟3秒执行");
            }
        }, 3000);

        Thread.sleep(5000);
        log.info("=============第二次启动定时器=============");
        timer.schedule(new TimerTask() {
            @Override
            public void run() {
                try {
                    Thread.sleep(3000);
                    log.info("周期性执行,启动定时器延迟3秒执行,之后每隔2秒执行一次,但如果任务时间超过2秒则以任务时间为间隔");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, 3000, 2000);
    }

在这里插入图片描述

scheduleAtFixedRate

	//将指定的任务安排为重复的固定速率执行,从指定的延迟后开始。
	//随后大约每隔一定的时间间隔进行,并按规定的时间间隔进行。
    public void scheduleAtFixedRate(TimerTask task, long delay, long period) {
        if (delay < 0)
            throw new IllegalArgumentException("Negative delay.");
        if (period <= 0)
            throw new IllegalArgumentException("Non-positive period.");
        sched(task, System.currentTimeMillis()+delay, period);
    }

    public void scheduleAtFixedRate(TimerTask task, Date firstTime,
                                    long period) {
        if (period <= 0)
            throw new IllegalArgumentException("Non-positive period.");
        sched(task, firstTime.getTime(), period);
    }
    public static void demo02() {
        Timer timer = new Timer();
        log.info("=============开始启动定时器=============");
        timer.scheduleAtFixedRate(new TimerTask() {
            @Override
            public void run() {
                try {
                    Thread.sleep(3000);
                    log.info("周期性执行,启动定时器延迟3秒执行,之后每隔2秒执行一次,但如果任务时间超过2秒则以任务时间为间隔");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }, 3000, 2000);
    }

在这里插入图片描述

2、ScheduledThreadPoolExecutor

    @Bean(name = "scheduledPoolTaskExecutor", destroyMethod = "shutdown")
    public ScheduledThreadPoolExecutor scheduledThreadPoolExecutor() {
        // 第一个参数:核心线程数
        // 第二个参数:线程工厂(此处用来设置线程名)
        // 第三次参数:饱和策略
        // 饱和策略
        // AbortPolicy:直接抛出java.util.concurrent.RejectedExecutionException异常
        // CallerRunsPolicy:若已达到待处理队列长度,将由主线程直接处理请求
        // DiscardOldestPolicy:抛弃旧的任务;会导致被丢弃的任务无法再次被执行
        // DiscardPolicy:抛弃当前任务;会导致被丢弃的任务无法再次被执行
        return new ScheduledThreadPoolExecutor(10,
                new ThreadFactory() {
                    int i = 0;

                    @Override
                    public Thread newThread(Runnable r) {
                        return new Thread(r, "scheduled-" + i++);
                    }
                },
                new ThreadPoolExecutor.DiscardPolicy()
        );
    }

@Slf4j
@Component
public class PoolTaskDemo {

    @Resource(name = "scheduledPoolTaskExecutor")
    private ScheduledThreadPoolExecutor scheduled;

    @PostConstruct
    public void scheduledPoolTaskExecutor() {
        schedule();
//        scheduleAtFixedRate();
//        scheduleWithFixedDelay();
    }


    public void schedule() {
        log.info("调用前");
        scheduled.schedule(new Runnable() {
            @Override
            public void run() {
                log.info("方法调用,延迟5秒执行一次, 只执行一次");
            }
        }, 5000, TimeUnit.MILLISECONDS);
    }
}

在这里插入图片描述

    public void scheduleAtFixedRate() {
        log.info("调用前");
        scheduled.scheduleAtFixedRate(new Runnable() {
            @SneakyThrows
            @Override
            public void run() {
                log.info("方法调用,延迟5秒首次执行,之后每过3秒轮询, 不受run方法执行耗时影响");
                Thread.sleep(2000);
            }
        }, 5000, 3000, TimeUnit.MILLISECONDS);
    }

在这里插入图片描述

    public void scheduleWithFixedDelay() {
        log.info("调用前");
        scheduled.scheduleWithFixedDelay(new Runnable() {
            @SneakyThrows
            @Override
            public void run() {
                log.info("方法调用,延迟5秒首次执行,之后每过3秒轮询,受run方法执行耗时影响");
                Thread.sleep(2000);
            }
        }, 5000, 3000, TimeUnit.MILLISECONDS);
    }

在这里插入图片描述

3、@Scheduled

@Configuration
@EnableScheduling // 使@Scheduled生效
public class ScheduledConfig implements SchedulingConfigurer {

    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        scheduledTaskRegistrar.setScheduler(scheduledThreadPoolExecutor());
    }

    @Bean(name = "scheduledPoolTaskExecutor", destroyMethod = "shutdown")
    public ScheduledThreadPoolExecutor scheduledThreadPoolExecutor() {
        // 第一个参数:核心线程数
        // 第二个参数:线程工厂(此处用来设置线程名)
        // 第三次参数:饱和策略
        // 饱和策略
        // AbortPolicy:直接抛出java.util.concurrent.RejectedExecutionException异常
        // CallerRunsPolicy:若已达到待处理队列长度,将由主线程直接处理请求
        // DiscardOldestPolicy:抛弃旧的任务;会导致被丢弃的任务无法再次被执行
        // DiscardPolicy:抛弃当前任务;会导致被丢弃的任务无法再次被执行
        return new ScheduledThreadPoolExecutor(10,
                new ThreadFactory() {
                    int i = 0;

                    @Override
                    public Thread newThread(Runnable r) {
                        return new Thread(r, "scheduled-" + i++);
                    }
                },
                new ThreadPoolExecutor.DiscardPolicy()
        );
    }

}

fixedRate

@Slf4j
@Component
public class ScheduledDemo {

    @Scheduled(fixedRate = 5000)
    public void fixedRate() throws InterruptedException {
        log.info("项目启动时立即执行第一次,之后每过5秒固定执行一次,不受方法执行耗时影响");
        Thread.sleep(2000);
    }
}    

在这里插入图片描述

fixedDelay

    @Scheduled(fixedDelay = 5000)
    public void fixedDelay() throws InterruptedException {
        log.info("项目启动时立即执行第一次,方法内部执行完成之后过5秒再次执行,受方法执行耗时影响");
        Thread.sleep(2000);
    }

在这里插入图片描述

initialDelay

    @Scheduled(initialDelay = 5000,fixedRate = 10000)
    public void initialDelayFixedRate() throws InterruptedException {
        log.info("项目启动时延迟5秒才执行第一次,在第一次执行后每过10秒固定执行一次,不受方法执行耗时影响");
        Thread.sleep(2000);
    }

在这里插入图片描述

cron

    @Scheduled(cron = "0/5 * * * * ? ")
    public void cron() throws InterruptedException {
        log.info("项目启动时延迟5秒才执行第一次,在第一次执行时每过5秒固定执行一次,不受方法执行耗时影响");
        Thread.sleep(2000);
    }

在这里插入图片描述

    @Scheduled(cron = "5 * * * * ? ")
    public void cron2() throws InterruptedException {
        log.info("每分钟的刚好第5秒时,执行一次,不受方法执行耗时影响");
        Thread.sleep(2000);
    }

在这里插入图片描述

cron表达式

cronExpression定义时间规则,Cron表达式由67个空格分隔的时间字段组成:秒 分钟 小时 日期 月份 星期 年(可选)


字段  允许值    允许的特殊字符 
秒       0-59     , - * / 
分       0-59     , - * / 
小时      0-23     , - * / 
日期      1-31     , - * ? / L W C 
月份      1-12     , - * / 
星期      1-7      , - * ? / L C # 
年     1970-2099     , - * /

特殊符号	代表含义
,	    枚举
-	    区间
*	    任意
/	    步长
?/星期冲突匹配
L	    最后
W	    工作日
CCalendar联系后计算过的值
#	    星期 4#22个星期四

如:
"*": 字符被用来指定所有的值。如:在分钟字段域里表示“每分钟”
"?":字符只在日期域和星期域中使用。它被用来指定“非明确的值”。
"-":字符被用来指定一个范围。如:“10-12”在小时域意味着“10点、11点、12点”。
",":字符被用来指定另外的值。如:“MON,WED,FRI”在星期域里表示”星期一、星期三、星期五”。
"/":字符用于指定增量。如:"0/5"在秒域0秒开始,每过5秒触发一次,"5/15"在分钟域表示每小时的5203550。
符号"*""/"前面(如:*/10)等价于0在“/”前面(如:0/10)。

L是‘last’的省略写法可以表示day-of-month和day-of-week域,但在两个字段中的意思不同,例如day-of- month域中表示一个月的最后一天。
如果在day-of-week域表示‘7’或者‘SAT’,如果在day-of-week域中前面加上数字,它表示 一个月的最后几天,例如‘6L’就表示一个月的最后一个星期五。
字符“W”只允许日期域出现。这个字符用于指定日期的最近工作日。例如:如果你在日期域中写 “15W”,表示:这个月15号最近的工作日。
所以,如果15号是周六,则任务会在14号触发。如果15好是周日,则任务会在周一也就是16号触发。
如果 是在日期域填写“1W”即使1号是周六,那么任务也只会在下周一,也就是3号触发,“W”字符指定的最近工作日是不能够跨月份的。
字符“W”只能配合一个 单独的数值使用,不能够是一个数字段,如:1-15W是错误的。
“L”和“W”可以在日期域中联合使用,LW表示这个月最后一周的工作日。
字符“#”只允许在星期域中出现。这个字符用于指定本月的某某天。例如:“6#3”表示本月第三周的星期五(6表示星期五,3表示第三周)。
“2#1”表示本月第一周的星期一。“4#5”表示第五周的星期三。
字符“C”允许在日期域和星期域出现。这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历” 关联,则等价于所有包含的“日历”。
如:日期域是“5C”表示关联“日历”中第一天,或者这个月开始的第一天的后5天。星期域是“1C”表示关联“日历” 中第一天,或者星期的第一天的后1天,也就是周日的后一天(周一)。

经典例子:
"0 0 10,14,16 * * ?"    每天上午10点,下午2点,4"0 0/30 9-17 * * ?"     朝九晚五工作时间内每半小时
"0 0 12 ? * WED"        表示每个星期三中午12"0 0 12 * * ?"          每天中午12点触发 
"0 15 10 ? * *"         每天上午10:15触发 
"0 15 10 * * ?"         每天上午10:15触发 
"0 15 10 * * ? *"       每天上午10:15触发 
"0 15 10 * * ? 2005"    2005年的每天上午10:15触发 
"0 * 14 * * ?"          在每天下午2点到下午2:59期间的每1分钟触发 
"0 0/5 14 * * ?"        在每天下午2点到下午2:55期间的每5分钟触发 
"0 0/5 14,18 * * ?"     在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发 
"0 0-5 14 * * ?"        在每天下午2点到下午2:05期间的每1分钟触发 
"0 10,44 14 ? 3 WED"    每年三月的星期三的下午2:102:44触发 
"0 15 10 ? * MON-FRI"   周一至周五的上午10:15触发 
"0 15 10 15 * ?"        每月15日上午10:15触发 
"0 15 10 L * ?"         每月最后一日的上午10:15触发 
"0 15 10 ? * 6L"        每月的最后一个星期五上午10:15触发 
"0 15 10 ? * 6#3"       每月的第三个星期五上午10:15触发

"30 * * * * ?"          每半分钟触发任务
"30 10 * * * ?"         每小时的1030秒触发任务
"30 10 1 * * ?"         每天11030秒触发任务
"30 10 1 20 * ?"        每月2011030秒触发任务
"30 10 1 20 10 ? *"     每年102011030秒触发任务
"30 10 1 20 10 ? 2011"  2011102011030秒触发任务
"30 10 1 ? 10 * 2011"   201110月每天11030秒触发任务
"30 10 1 ? 10 SUN 2011" 201110月每周日11030秒触发任务
"15,30,45 * * * * ?"    15秒,30秒,45秒时触发任务
"15-45 * * * * ?"       1545秒内,每秒都触发任务
"15/5 * * * * ?"        每分钟的每15秒开始触发,每隔5秒触发一次
"15-30/5 * * * * ?"     每分钟的15秒到30秒之间开始触发,每隔5秒触发一次
"0 0/3 * * * ?"         每小时的第00秒开始,每三分钟触发一次
"0 15 10 ? * MON-FRI"   星期一到星期五的10150秒触发任务
"0 15 10 L * ?"         每个月最后一天的10150秒触发任务
"0 15 10 LW * ?"        每个月最后一个工作日的10150秒触发任务
"0 15 10 ? * 5L"        每个月最后一个星期四的10150秒触发任务
"0 15 10 ? * 5#3"       每个月第三周的星期四的10150秒触发任务
  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值