在日常项目运行中,我们总会有需求在某一时间段周期性的执行某个动作。比如每天在某个时间段导出报表,或者每隔多久统计一次现在在线的用户量。在springboot中可以有很多方案去帮我们完成定时器的工作,有Java自带的java.util.Timer类,也有强大的调度器Quartz,还有SpringBoot自带的Scheduled,今天主要说说Scheduled。

一、定时器比较

框架名称 Cron表达式 固定间隔执行 固定频率执行 任务持久化 难易度

TimerTask

不支持 支持 支持 不支持 一般

schedule

支持 支持 支持 不支持 简单

Quartz

支持 支持 支持 支持

在实际应用中,如果没有分布式场景(quartz 支持分布式, schedule 不支持(需要自己实现,用分布式锁,哪个拿到了这把锁,哪个就行执行),schedule跟spring结合的更好,还是很适用的。

二、SpringBoot的定时任务

使用SpringBoot创建定时任务非常简单,目前主要有以下三种创建方式:

  • 基于注解
  • 基于SchedulingConfigurer接口,我们可以从数据库中读取指定时间来动态执行定时任务
  • 基于注解设定多线程定时任务

1. 基于注解@Scheduled

注解@Scheduled默认为单线程,开启多个任务时,任务的执行时机会受上一个任务执行时间的影响。即多个任务都是在同个线程中先后执行。

  1. @Configuration //标识为配置类
  2. @EnableScheduling // 开启定时任务
  3. public class ScheduleTask {
  4. //3.添加定时任务
  5. @Scheduled(cron = "0/5 * * * * ?")
  6. //或直接指定时间间隔,例如:5秒
  7. //@Scheduled(fixedRate=5000)
  8. private void configureTasks() {
  9. System.err.println("执行静态定时任务时间: " + LocalDateTime.now());
  10. }
  11. }

@Scheduled除了支持灵活的参数表达式cron之外,还支持简单的延时操作,例如 fixedDelay ,fixedRate 填写相应的毫秒数即可。

cron表达式详解:https://www.cnblogs.com/myitnews/p/11863041.html

2. 基于SchedulingConfigurer接口

@Schedule注解的一个缺点就是其定时时间不能动态更改,它适用于具有固定任务周期的任务,若要修改任务执行周期,只能走“停服务→修改任务执行周期→重启服务”这条路。而基于 SchedulingConfigurer 接口方式可以做到。SchedulingConfigurer 接口可以实现在@Configuration 类上,同时不要忘了,还需要@EnableScheduling 注解的支持。

接口源码:

  1. @FunctionalInterface
  2. public interface SchedulingConfigurer {
  3. void configureTasks(ScheduledTaskRegistrar var1);
  4. }

ScheduledTaskRegistrar类包括以下几个重要方法:

  • void addTriggerTask(Runnable task, Trigger trigger)
  • void addTriggerTask(TriggerTask task)
  • void addCronTask(Runnable task, String expression)
  • void addCronTask(CronTask task)
  • void addFixedRateTask(Runnable task, long interval)
  • void addFixedRateTask(IntervalTask task)
  • void addFixedDelayTask(Runnable task, long delay)
  • void addFixedDelayTask(IntervalTask task)
  1. @Component
  2. @EnableScheduling
  3. public class TestTask implements SchedulingConfigurer {
  4.  
  5. @Override
  6. public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
  7. taskRegistrar.addTriggerTask(new Runnable() {
  8. @Override
  9. public void run() {
  10. // 定时任务要执行的内容
  11. System.out.println("【开始执行定时任务。。。】");
  12. }
  13. }, new Trigger() {
  14. @Override
  15. public Date nextExecutionTime(TriggerContext triggerContext) {
  16. // 定时任务触发,可修改定时任务的执行周期
  17. String cron = "0 0/5 * * * ?"; //可以将表达式配置在数据库中
  18. CronTrigger trigger = new CronTrigger(cron);
  19. Date nextExecDate = trigger.nextExecutionTime(triggerContext);
  20. return nextExecDate;
  21. }
  22. });
  23. }
  24. }

注意: 如果在数据库修改时格式出现错误,则定时任务会停止,即使重新修改正确;此时只能重新启动项目才能恢复。

如果确实可能存在数据库表达式错误的,可以是使用org.springframework.scheduling.support.CronSequenceGenerator的isValidExpression验证下,错误的给个默认执行表达式。

3. 多线程定时任务

多线程定时任务是基于@Async注解的。

  1. @Component
  2. @EnableScheduling // 1.开启定时任务
  3. @EnableAsync // 2.开启多线程
  4. public class MultithreadScheduleTask {
  5.  
  6. @Async
  7. @Scheduled(fixedDelay = 1000) //间隔1秒
  8. public void first() throws InterruptedException {
  9. System.out.println("第一个定时任务开始 : " + LocalDateTime.now().toLocalTime() + "\r\n线程 : " + Thread.currentThread().getName());
  10. System.out.println();
  11. Thread.sleep(1000 * 10);
  12. }
  13.  
  14. @Async
  15. @Scheduled(fixedDelay = 2000)
  16. public void second() {
  17. System.out.println("第二个定时任务开始 : " + LocalDateTime.now().toLocalTime() + "\r\n线程 : " + Thread.currentThread().getName());
  18. System.out.println();
  19. }
  20. }

第一个定时任务和第二个定时任务互不影响;由于开启了多线程,第一个任务的执行时间也不受其本身执行时间的限制,所以需要注意可能会出现重复操作导致数据异常。

三、分布式环境下的定时任务

我们有很多时候都需要一些定时任务的辅助,大多数情况,定时任务都可能是写到一个服务节点。但是可能存在以下情况:

  • 如果业务逻辑过于复杂的话,不好维护。
  • 如果服务节点挂了,那么所有的定时任务都不会执行了。

可能有人说部署多个节点不久可以了吗?确实,但是有没有想过,定时任务会重复执行,这怎么解决呢?使用redis分布式锁。

查看本教程中的 “RedisTemplate详解” ,获取RedisUtil的代码。

  1. boolean flag = false;
  2. try {
  3. //1. 获取锁
  4. flag = RedisUtil.lock("smsSend", 1000); //key和过期时间,过期时间根据实际业务预测
  5. if(flag) {
  6. //2. 执行业务逻辑
  7. }else {
  8. logger.info("分布式锁没有获取到锁:{}");
  9. }
  10. } catch (Exception e) {
  11. logger.info("定时任务执行失败,分布式锁异常"+e);
  12. }finally {
  13. if(flag) {//释放锁
  14. RedisUtil.unlock("smsSend");
  15. logger.info("锁已释放");
  16. }
  17. }

【使用篇二】SpringBoot定时任务Scheduled(14)的更多相关文章

  1. SpringBoot学习笔记(七):SpringBoot使用AOP统一处理请求日志、SpringBoot定时任务@Scheduled、SpringBoot异步调用Async、自定义参数

    SpringBoot使用AOP统一处理请求日志 这里就提到了我们Spring当中的AOP,也就是面向切面编程,今天我们使用AOP去对我们的所有请求进行一个统一处理.首先在pom.xml中引入我们需要的 ...

  2. SpringBoot定时任务@Scheduled

    SpringBoot定时任务主要由两个注解完成. @Scheduled加在方法上面. @EnableScheduling加在类上面.可以是Application类,也可以是@Component类,还可 ...

  3. Springboot定时任务@Scheduled注解形式,参数详解

    参数详解 1.占位符 1 秒 是 0-59 , - * / 2 分 是 0-59 , - * / 3 时 是 0-23 , - * / 4 日 是 1-31 , - * ? / L W 5 月 是 1 ...

  4. spring boot 学习(八)定时任务 @Scheduled

    SpringBoot 定时任务 @Scheduled 前言 有时候,我们有这样的需求,需要在每天的某个固定时间或者每隔一段时间让应用去执行某一个任务.一般情况下,可以使用多线程来实现这个功能:在 Sp ...

  5. SpringBoot学习18:springboot使用Scheduled 定时任务器

    Scheduled 定时任务器:是 Spring3.0 以后自带的一个定时任务器. 1.在pom.xml文件中添加Scheduled依赖 <!-- 添加spring定时任务 Scheduled ...

  6. SpringBoot 定时任务篇

    一. 基于注解@Scheduled默认为单线程,开启多个任务时,任务的执行时机会受上一个任务执行时间的影响. 1.创建定时器 使用SpringBoot基于注解来创建定时任务非常简单,只需几行代码便可完 ...

  7. springboot 基于@Scheduled注解 实现定时任务

    前言 使用SpringBoot创建定时任务非常简单,目前主要有以下三种创建方式: 一.基于注解(@Scheduled) 二.基于接口(SchedulingConfigurer) 前者相信大家都很熟悉, ...

  8. 14 微服务电商【黑马乐优商城】:day02-springcloud(理论篇二:知道什么是SpringCloud)

    本项目的笔记和资料的Download,请点击这一句话自行获取. day01-springboot(理论篇) :day01-springboot(实践篇) day02-springcloud(理论篇一: ...

  9. SpringBoot执行定时任务@Scheduled

    SpringBoot执行定时任务@Scheduled 在做项目时,需要一个定时任务来接收数据存入数据库,后端再写一个接口来提供该该数据的最新的那一条. 数据保持最新:设计字段sign的值(0,1)来设 ...

随机推荐

  1. Golang 在电商即时通讯服务建设中的实践

    马蜂窝技术原创文章,更多干货请搜索公众号:mfwtech ​即时通讯(IM)功能对于电商平台来说非常重要,特别是旅游电商. 从商品复杂性来看,一个旅游商品可能会包括用户在未来一段时间的衣.食.住.行等 ...

  2. 同时安装Office2016和Visio2016

    最近由于学习需要使用Visio,于是下载来安装.之前使用的Office2016一切正常,但是怎么都安装不上visio,把Office2016卸载了安,又没有word.ppt这些,反正就是没法同时安装上 ...

  3. robot_framework常用关键字

    快捷键 F8 运行 ctrl+alt+空格 log 类似于print Set variable 定义变量 Catenate 连接对象 SEPARATOR 对多个连接信息进行分割 Create List ...

  4. redis(4)--redis集群之主从复制

    集群 先来简单了解下redis中提供的集群策略, 虽然redis有持久化功能能够保障redis服务器宕机也能恢复并且只有少量的数据损失,但是由于所有数据在一台服务器上,如果这台服务器出现硬盘故障,那就 ...

  5. 一次框架性能的比较,引起了我对搭建web框架的兴趣

    背景 一次无意的访问,点击到了一个专门做PHP性能测试的网站,看这里PHP Benchmarks. 在里面发现了框架性能测试的结果,发现Laravel的框架性能尽然是最低的.瞬间受到了一万点的暴击,谁 ...

  6. 这道Java基础题真的有坑!我求求你,认真思考后再回答。

    本文目录 一.题是什么题? 二.阿里Java开发规范. 2.1 正例代码. 2.2 反例代码. 三.层层揭秘,为什么发生异常了呢? 3.1 第一层:异常信息解读. 3.2 第二层:抛出异常的条件解读. ...

  7. Asp.Net MVC中Aplayer.js音乐播放器的使用

    1.前言: Aplater.js是一款可爱.漂亮的Js音乐播放器,以前就了解过也弄过一些,现在就用mp3的格式来在.Net里面开发.管网 https://aplayer.js.org/ 2.入手: 在 ...

  8. visualStudio 的一些常用使用操作总结

    今年苟了差不多一整年,期间断断续续把c++ prime plus 看完了 ,发现并没有什么鸟用 ,但是对代码怎么形成二进制的过程 动态内存管理 这些模模糊糊的确实理解更深刻些了 特别是c++过度到c# ...

  9. 记录AJAX充电点点滴滴

    首先要明白什么是 AJAX ? AJAX = 异步 JavaScript 和 XML. AJAX 是一种用于创建快速动态网页的技术. 通过在后台与服务器进行少量数据交换,AJAX 可以使网页实现异步更 ...

  10. Ansible-playbook之循环判断

    1.循环 (loop) # 使用循环创建硬连接:x连接到y:z连接到k: - hosts: web - name: Create two hard links file: src: "{{ ...