百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

你需要的spring定时任务讲解+案例都在这里

yuyutoo 2024-10-12 00:07 2 浏览 0 评论

本文目录

  • 定时任务概述、原理
  • 基于注解(@Scheduled)的定时任务
  • 基于接口(SchedulingConfigurer)的定时任务
  • 可更改时间的定时任务

一、定时任务概述、原理

概述:开发中经常会使用到定时任务,顾名思义,定时任务就是定时执行的方法,即定时执行的代码。比如,为了减少服务器或者数据库的压力,我们会将一些对服务器或者数据库等有压力的高频操作,改为定时去执行,例如每晚凌晨0点同步A系统的数据到B系统,每2小时统计用户的积分情况,每周一给支付宝用户推送上周收入支出数据报表等。一 般情况下,很多业务处理会定时在凌晨处理,因为避开了户使用高峰期,服务器资源充足,而且对用户影响小。

原理:spring在初始化bean后,通过“postProcessAfterInitialization”拦截到所有的用到“@Scheduled”注解的方法,并解析相应的的注解参数,放入“定时任务列表”等待后续处理;之后再“定时任务列表”中统一执行相应的定时任务(任务为顺序执行,先执行cron,之后再执行fixedRate)。

二、基于注解(@Scheduled)的定时任务

  需要注意的是@Scheduled默认是串行的,单线程,当开启多个任务时,任务的执行时机会受上一个任务执行时间的影响。

@Configuration

@EnableScheduling //开启定时任务

public class ScheduleTask {

//每10秒执行一次

@Scheduled(cron = "0/10 * * * * ?")

private void configureTasks() {

System.out.println("我是一个定时任务");

}

}

@Scheduled 除了cron还提供另外三种种方式: fixedRate,fixedDelay,initialDelay

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

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

@Configuration

@EnableScheduling //开启定时任务

public class ScheduleTask {

//每10秒执行一次

@Scheduled(fixedDelay = 10000)

private void configureTasks() {

System.out.println("我是一个定时任务");

}

}

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

@Configuration

@EnableScheduling //开启定时任务

public class ScheduleTask {

//每10秒执行一次

@Scheduled(fixedRate = 10000)

private void configureTasks() {

System.out.println("我是一个定时任务");

}

}

  4、initialDelay = 10000 表示在容器启动后,延迟10秒后再执行一次定时器。

@Configuration

@EnableScheduling //开启定时任务

public class ScheduleTask {

//容器启动后,延迟10秒后再执行一次定时器,以后每10秒再执行一次该定时器。

@Scheduled(initialDelay = 10000, fixedRate = 10000)

private void configureTasks() {

System.out.println("我是一个定时任务");

}

}

三、基于接口(SchedulingConfigurer)的定时任务

  有些程序员可能发现,使用@Scheduled 注解很方便,但缺点是当我们调整了执行周期的时候,需要重启应用才能生效,这多少有些不方便。为了达到实时生效的效果,那么可以使用接口来完成定时任务。

  下面案例从数据中获取执行周期时间,然后动态执行定时任务:

@Configuration

@EnableScheduling //开启定时任务

public class DynamicScheduleTask implements SchedulingConfigurer {

//从数据获取任务执行周期

@Autowired

private MyBatisMapper myBatisMapper;

@Override

public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {

taskRegistrar.addTriggerTask(

//1.添加任务内容(Runnable)

() -> System.out.println("执行动态定时任务: " + LocalDateTime.now().toLocalTime()),

//2.设置执行周期(Trigger)

triggerContext -> {

//2.1从数据库获取执行周期

String cron = myBatisMapper.getCron();

//2.2 返回执行周期(Date)

return new CronTrigger(cron).nextExecutionTime(triggerContext);

}

);

}

}

数据库表数据如下:

现在,我们启动测试看看

执行动态定时任务: 17:17:00.008999

执行动态定时任务: 17:17:20.002501

执行动态定时任务: 17:17:30.001786

执行动态定时任务: 17:17:40.005512

执行动态定时任务: 17:17:50.005870

执行动态定时任务: 17:18:00.002189

执行动态定时任务: 17:18:10.001910

我们可以看到每10秒执行一次任务。那么现在要求每5秒执行一次,该怎么做呢?这个时候我们只需要修改下数据库数据即可,无需重启。

现在,我们再看看控制台打印的是什么?

执行动态定时任务: 17:18:30.000902

执行动态定时任务: 17:18:40.001392

执行动态定时任务: 17:18:45.005027

执行动态定时任务: 17:18:50.001367

执行动态定时任务: 17:18:55.001356

执行动态定时任务: 17:19:00.001582

执行动态定时任务: 17:19:05.005676

执行动态定时任务: 17:19:10.001258

执行动态定时任务: 17:19:15.005272

成功每5秒执行一次。是不是很有成就感,哈哈

四、可更改时间的定时任务

案例如下:

  package com.nobody.task;

  import org.slf4j.Logger;

  import org.slf4j.LoggerFactory;

  import org.springframework.scheduling.Trigger;

  import org.springframework.scheduling.annotation.SchedulingConfigurer;

  import org.springframework.scheduling.config.ScheduledTaskRegistrar;

  import org.springframework.scheduling.support.CronTrigger;

  import org.springframework.stereotype.Component;

  /**

  * @Description 可动态更改时间的定时任务

  * @Author Mr.nobody

  * @Date 2021/3/4

  * @Version 1.0.0

  */

  @Component

  public class ChangeTimeScheduledTask implements SchedulingConfigurer {

  private static final Logger LOGGER = LoggerFactory.getLogger(ChangeTimeScheduledTask.class);

  // cron表达式,我们动态更改此属性的值即可更改定时任务的执行时间

  private String expression = "0/5 * * * * *";

  @Override

  public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {

  // 定时任务要执行的方法

  Runnable task = () -> LOGGER.info(">>> configureTasks ...");

  // 调度实现的时间控制

  Trigger trigger = triggerContext -> {

  CronTrigger cronTrigger = new CronTrigger(expression);

  return cronTrigger.nextExecutionTime(triggerContext);

  };

  taskRegistrar.addTriggerTask(task, trigger);

  }

  public String getExpression() {

  return expression;

  }

  public void setExpression(String expression) {

  this.expression = expression;

  }

  }

接口调用:

  package com.nobody.controller;

  import com.nobody.task.ChangeTimeScheduledTask;

  import org.springframework.web.bind.annotation.GetMapping;

  import org.springframework.web.bind.annotation.RequestMapping;

  import org.springframework.web.bind.annotation.RestController;

  /**

  * @Description

  * @Author Mr.nobody

  * @Date 2021/3/4

  * @Version 1.0.0

  */

  @RestController

  @RequestMapping("demo")

  public class DemoController {

  private ChangeTimeScheduledTask changeTimeScheduledTask;

  public DemoController(final ChangeTimeScheduledTask changeTimeScheduledTask) {

  this.changeTimeScheduledTask = changeTimeScheduledTask;

  }

  @GetMapping

  public String testChangeTimeScheduledTask() {

  changeTimeScheduledTask.setExpression("0/10 * * * * *");

  return "ok";

  }

  }

  启动服务,没调用接口之前,定时任务是每5秒执行一次

  2021-03-04 14:02:10.001 INFO 6836 --- [TaskScheduler-1] com.nobody.task.ChangeTimeScheduledTask : >>> configureTasks ...

  2021-03-04 14:02:15.001 INFO 6836 --- [TaskScheduler-1] com.nobody.task.ChangeTimeScheduledTask : >>> configureTasks ...

  2021-03-04 14:02:20.002 INFO 6836 --- [TaskScheduler-1] com.nobody.task.ChangeTimeScheduledTask : >>> configureTasks ...

  2021-03-04 14:02:25.001 INFO 6836 --- [TaskScheduler-1] com.nobody.task.ChangeTimeScheduledTask : >>> configureTasks ...

  然后我们调用接口,改变定时任务的时间,结果变为每10秒执行一次。

  2021-03-04 14:02:30.005 INFO 6836 --- [TaskScheduler-1] com.nobody.task.ChangeTimeScheduledTask : >>> configureTasks ...

  2021-03-04 14:02:35.002 INFO 6836 --- [TaskScheduler-1] com.nobody.task.ChangeTimeScheduledTask : >>> configureTasks ...

  2021-03-04 14:02:40.001 INFO 6836 --- [TaskScheduler-1] com.nobody.task.ChangeTimeScheduledTask : >>> configureTasks ...

完!

相关推荐

jQuery VS AngularJS 你更钟爱哪个?

在这一次的Web开发教程中,我会尽力解答有关于jQuery和AngularJS的两个非常常见的问题,即jQuery和AngularJS之间的区别是什么?也就是说jQueryVSAngularJS?...

Jquery实时校验,指定长度的「负小数」,小数位未满末尾补0

在可以输入【负小数】的输入框获取到焦点时,移除千位分隔符,在输入数据时,实时校验输入内容是否正确,失去焦点后,添加千位分隔符格式化数字。同时小数位未满时末尾补0。HTML代码...

如何在pbootCMS前台调用自定义表单?pbootCMS自定义调用代码示例

要在pbootCMS前台调用自定义表单,您需要在后台创建表单并为其添加字段,然后在前台模板文件中添加相关代码,如提交按钮和表单验证代码。您还可以自定义表单数据的存储位置、添加文件上传字段、日期选择器、...

编程技巧:Jquery实时验证,指定长度的「负小数」

为了保障【负小数】的正确性,做成了通过Jquery,在用户端,实时验证指定长度的【负小数】的方法。HTML代码<inputtype="text"class="forc...

一篇文章带你用jquery mobile设计颜色拾取器

【一、项目背景】现实生活中,我们经常会遇到配色的问题,这个时候去百度一下RGB表。而RGB表只提供相对于的颜色的RGB值而没有可以验证的模块。我们可以通过jquerymobile去设计颜色的拾取器...

编程技巧:Jquery实时验证,指定长度的「正小数」

为了保障【正小数】的正确性,做成了通过Jquery,在用户端,实时验证指定长度的【正小数】的方法。HTML做成方法<inputtype="text"class="fo...

jquery.validate检查数组全部验证

问题:html中有多个name[],每个参数都要进行验证是否为空,这个时候直接用required:true话,不能全部验证,只要这个数组中有一个有值就可以通过的。解决方法使用addmethod...

Vue进阶(幺叁肆):npm查看包版本信息

第一种方式npmviewjqueryversions这种方式可以查看npm服务器上所有的...

layui中使用lay-verify进行条件校验

一、layui的校验很简单,主要有以下步骤:1.在form表单内加上class="layui-form"2.在提交按钮上加上lay-submit3.在想要校验的标签,加上lay-...

jQuery是什么?如何使用? jquery是什么功能组件

jQuery于2006年1月由JohnResig在BarCampNYC首次发布。它目前由TimmyWilson领导,并由一组开发人员维护。jQuery是一个JavaScript库,它简化了客户...

django框架的表单form的理解和用法-9

表单呈现...

jquery对上传文件的检测判断 jquery实现文件上传

总体思路:在前端使用jquery对上传文件做部分初步的判断,验证通过的文件利用ajaxFileUpload上传到服务器端,并将文件的存储路径保存到数据库。<asp:FileUploadI...

Nodejs之MEAN栈开发(四)-- form验证及图片上传

这一节增加推荐图书的提交和删除功能,来学习node的form提交以及node的图片上传功能。开始之前需要源码同学可以先在git上fork:https://github.com/stoneniqiu/R...

大数据开发基础之JAVA jquery 大数据java实战

上一篇我们讲解了JAVAscript的基础知识、特点及基本语法以及组成及基本用途,本期就给大家带来了JAVAweb的第二个知识点jquery,大数据开发基础之JAVAjquery,这是本篇文章的主要...

推荐四个开源的jQuery可视化表单设计器

jquery开源在线表单拖拉设计器formBuilder(推荐)jQueryformBuilder是一个开源的WEB在线html表单设计器,开发人员可以通过拖拉实现一个可视化的表单。支持表单常用控件...

取消回复欢迎 发表评论: