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

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

yuyutoo 2024-10-12 00:07 10 浏览 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 ...

完!

相关推荐

Python操作Word文档神器:python-docx库从入门到精通

Python操作Word文档神器:python-docx库从入门到精通动动小手,点击关注...

Python 函数调用从入门到精通:超详细定义解析与实战指南 附案例

一、函数基础:定义与调用的核心逻辑定义:函数是将重复或相关的代码块封装成可复用的单元,通过函数名和参数实现特定功能。它是Python模块化编程的基础,能提高代码复用性和可读性。定义语法:...

等这么长时间Python背记手册终于来了,入门到精通(视频400集)

本文毫无套路!真诚分享!前言:无论是学习任何一门语言,基础知识一定要扎实,基础功非常的重要,找一个有丰富编程经验的老师或者师兄带着你会少走很多弯路,你的进步速度也会快很多,无论我们学习的目的是什么,...

图解Python编程:从入门到精通系列教程(附全套速查表)

引言本系列教程展开讲解Python编程语言,Python是一门开源免费、通用型的脚本编程语言,它上手简单,功能强大,它也是互联网最热门的编程语言之一。Python生态丰富,库(模块)极其丰富,这使...

Python入门教程(非常详细)从零基础入门到精通,看完这一篇就够

本书是Python经典实例解析,采用基于实例的方法编写,每个实例都会解决具体的问题和难题。主要内容有:数字、字符串和元组,语句与语法,函数定义,列表、集、字典,用户输入和输出等内置数据结构,类和对象,...

Python函数全解析:从入门到精通,一文搞定!

1.为什么要用函数?函数的作用:封装代码,提高复用性,减少重复,提高可读性。...

Python中的单例模式:从入门到精通

Python中的单例模式:从入门到精通引言单例模式是一种常用的软件设计模式,它保证了一个类只有一个实例,并提供一个全局访问点。这种模式通常用于那些需要频繁创建和销毁的对象,比如日志对象、线程池、缓存等...

【Python王者归来】手把手教你,Python从入门到精通!

用800个程序实例、5万行代码手把手教你,Python从入门到精通!...

Python从零基础入门到精通:一个月就够了

如果想从零基础到入门,能够全职学习(自学),那么一个月足够了。...

Python 从入门到精通:一个月就够了

要知道,一个月是一段很长的时间。如果每天坚持用6-7小时来做一件事,你会有意想不到的收获。作为初学者,第一个月的月目标应该是这样的:熟悉基本概念(变量,条件,列表,循环,函数)练习超过30个编...

Python零基础到精通,这8个入门技巧让你少走弯路,7天速通编程!

Python学习就像玩积木,从最基础的块开始,一步步搭建出复杂的作品。我记得刚开始学Python时也是一头雾水,走了不少弯路。现在回头看,其实掌握几个核心概念,就能快速入门这门编程语言。来聊聊怎么用最...

神仙级python入门教程(非常详细),从0到精通,从看这篇开始!

python入门虽然简单,很多新手依然卡在基础安装阶段,大部分教程对一些基础内容都是一带而过,好多新手朋友,对一些基础知识常常一知半解,需要在网上查询很久。...

Python类从入门到精通,一篇就够!

一、Python类是什么?大家在生活中应该都见过汽车吧,每一辆真实存在、能在路上跑的汽车,都可以看作是一个“对象”。那这些汽车是怎么生产出来的呢?其实,在生产之前,汽车公司都会先设计一个详细的蓝图...

学习Python从入门到精通:30天足够了,这才是python基础的天花板

当年2w买的全套python教程用不着了,现在送给有缘人,不要钱,一个月教你从入门到精通1、本套视频共487集,本套视频共分4季...

30天Python 入门到精通(3天学会python)

以下是一个为期30天的Python入门到精通学习课程,专为零基础新手设计。课程从基础语法开始,逐步深入到面向对象编程、数据处理,最后实现运行简单的大语言模型(如基于HuggingFace...

取消回复欢迎 发表评论: