使用@Schedule注解实现定时任务,多线程执行定时任务,Cron表达式详解

news2024/11/27 14:44:06

@Schedule注解实现定时任务,多线程执行定时任务,Cron表达式详解

    • 使用@Schedule注解实现定时任务
    • @Scheduled注解
    • 多线程执行定时任务
    • Cron表达式
      • Cron中的通配符

使用@Schedule注解实现定时任务

1、首先,在项目启动类上添加 @EnableScheduling 注解,开启对定时任务的支持。@EnableScheduling 注解的作用是发现注解 @Scheduled 的任务并在后台执行该任务。

@SpringBootApplication
@EnableScheduling
public class ScheduledApplication {

	public static void main(String[] args) {
		SpringApplication.run(ScheduledApplication.class, args);
	}

}

2、编写定时任务类和方法,定时任务类通过 Spring IOC 加载,使用 @Component 注解。
3、定时方法使用 @Scheduled注解。下述代码中,fixedRate 是 long 类型,表示任务执行的间隔毫秒数,下面代码中的定时任务每 3 秒执行一次。

@Component
public class ScheduledTask {

    @Scheduled(fixedRate = 3000)
    public void scheduledTask() {
        System.out.println("任务执行时间:" + LocalDateTime.now());
    }

}

4、运行工程,项目启动和运行日志如下,可见每 3 秒打印一次日志执行记录。

Server is running ...
任务执行时间-ScheduledTask2020-06-23T18:02:14.747
任务执行时间-ScheduledTask2020-06-23T18:02:17.748
任务执行时间-ScheduledTask2020-06-23T18:02:20.746
任务执行时间-ScheduledTask2020-06-23T18:02:23.747

@Scheduled注解

在上面 Demo 中,使用了 @Scheduled(fixedRate = 3000) 注解来定义每过 3 秒执行的任务。对于 @Scheduled 的使用可以总结如下几种方式。

@Scheduled(fixedRate = 3000) 上一次开始执行时间点之后 3 秒再执行(fixedRate
属性:定时任务开始后再次执行定时任务的延时(需等待上次定时任务完成),单位毫秒)。

@Scheduled(fixedDelay = 3000) 上一次执行完毕时间点之后 3 秒再执行(fixedDelay
属性:定时任务执行完成后再次执行定时任务的延时(需等待上次定时任务完成),单位毫秒)。

@Scheduled(initialDelay = 1000, fixedRate = 3000) 第一次延迟1秒后执行,之后按
fixedRate 的规则每 3 秒执行一次( initialDelay 属性:第一次执行定时任务的延迟时间,需配合 fixedDelay
或者 fixedRate 来使用)。

@Scheduled(cron="0 0 2 1 * ? * ") 通过 cron 表达式定义规则。

关于「Corn表达式」,将在文末介绍。

多线程执行定时任务

import org.slf4j.LoggerFactory;

@Component
public class ScheduledTask {
    private static final org.slf4j.Logger logger = LoggerFactory.getLogger(ScheduledTask.class);

    @Scheduled(cron = "0/5 * * * * *")
    public void scheduled(){
        logger.info("使用cron---任务执行时间:{}  线程名称:{}",LocalDateTime.now(),Thread.currentThread().getName());
    }
    
    @Scheduled(fixedRate = 5000)
    public void scheduled1() {
        logger.info("fixedRate---任务执行时间:{}  线程名称:{}",LocalDateTime.now(),Thread.currentThread().getName());
    }
    @Scheduled(fixedDelay = 5000)
    public void scheduled2() {
        logger.info("fixedDelay---任务执行时间:{}  线程名称:{}",LocalDateTime.now(),Thread.currentThread().getName());
    }
}

程序输出如下。

2020-06-23 23:31:04.447  INFO 34069 : fixedRate---任务执行时间:2020-06-23T23:31:04.447  线程名称:scheduling-1
2020-06-23 23:31:04.494  INFO 34069 : fixedDelay---任务执行时间:2020-06-23T23:31:04.494  线程名称:scheduling-1
2020-06-23 23:31:05.004  INFO 34069 : 使用cron---任务执行时间:2020-06-23T23:31:05.004  线程名称:scheduling-1
2020-06-23 23:31:09.445  INFO 34069 : fixedRate---任务执行时间:2020-06-23T23:31:09.445  线程名称:scheduling-1
2020-06-23 23:31:09.498  INFO 34069 : fixedDelay---任务执行时间:2020-06-23T23:31:09.498  线程名称:scheduling-1
2020-06-23 23:31:10.003  INFO 34069 : 使用cron---任务执行时间:2020-06-23T23:31:10.003  线程名称:scheduling-1
2020-06-23 23:31:14.444  INFO 34069 : fixedRate---任务执行时间:2020-06-23T23:31:14.444  线程名称:scheduling-1
2020-06-23 23:31:14.503  INFO 34069 : fixedDelay---任务执行时间:2020-06-23T23:31:14.503  线程名称:scheduling-1
2020-06-23 23:31:15.002  INFO 34069 : 使用cron---任务执行时间:2020-06-23T23:31:15.002  线程名称:scheduling-1

可以看到,3 个定时任务都已经执行,并且使同一个线程中串行执行。当定时任务增多,如果一个任务卡死,会导致其他任务也无法执行。

因此,需要考虑多线程执行定时任务的情况。

1、创建配置类:在传统的 Spring 项目中,我们可以在 xml 配置文件添加 task 的配置,而在 Spring Boot 项目中一般使用 config 配置类的方式添加配置,所以新建一个 AsyncConfig 类。在配置类中,使用 @EnableAsync 注解开启异步事件的支持。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.util.concurrent.Executor;

@Configuration
@EnableAsync

public class AsyncConfig {

    private int corePoolSize = 10;
    private int maxPoolSize = 200;
    private int queueCapacity = 10;

    @Bean
    public Executor taskExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(corePoolSize);
        executor.setMaxPoolSize(maxPoolSize);
        executor.setQueueCapacity(queueCapacity);
        executor.initialize();
        return executor;
    }
}

@Configuration:表明该类是一个配置类
@EnableAsync:开启异步事件的支持

2、在定时任务的类或者方法上添加 @Async 注解,表示是异步事件的定时任务。

@Component
@Async
public class ScheduledTask {
    private static final org.slf4j.Logger logger = LoggerFactory.getLogger(ScheduledTask.class);
    

    @Scheduled(cron = "0/5 * * * * *")
    public void scheduled(){
        logger.info("使用cron  线程名称:{}",Thread.currentThread().getName());
    }
    
    @Scheduled(fixedRate = 5000)
    public void scheduled1() {
        logger.info("fixedRate--- 线程名称:{}",Thread.currentThread().getName());
    }
    @Scheduled(fixedDelay = 5000)
    public void scheduled2() {
        logger.info("fixedDelay  线程名称:{}",Thread.currentThread().getName());
    }
}

3、运行程序,控制台输出如下,可以看到,定时任务是在多个线程中执行的。

2020-06-23 23:45:08.514  INFO 34824 : fixedRate--- 线程名称:taskExecutor-1
2020-06-23 23:45:08.514  INFO 34824 : fixedDelay  线程名称:taskExecutor-2
2020-06-23 23:45:10.005  INFO 34824 : 使用cron  线程名称:taskExecutor-3

2020-06-23 23:45:13.506  INFO 34824 : fixedRate--- 线程名称:taskExecutor-4
2020-06-23 23:45:13.510  INFO 34824 : fixedDelay  线程名称:taskExecutor-5
2020-06-23 23:45:15.005  INFO 34824 : 使用cron  线程名称:taskExecutor-6

2020-06-23 23:45:18.509  INFO 34824 : fixedRate--- 线程名称:taskExecutor-7
2020-06-23 23:45:18.511  INFO 34824 : fixedDelay  线程名称:taskExecutor-8
2020-06-23 23:45:20.005  INFO 34824 : 使用cron  线程名称:taskExecutor-9

2020-06-23 23:45:23.509  INFO 34824 : fixedRate--- 线程名称:taskExecutor-10
2020-06-23 23:45:23.511  INFO 34824 : fixedDelay  线程名称:taskExecutor-1
2020-06-23 23:45:25.005  INFO 34824 : 使用cron  线程名称:taskExecutor-2

2020-06-23 23:45:28.509  INFO 34824 : fixedRate--- 线程名称:taskExecutor-3
2020-06-23 23:45:28.512  INFO 34824 : fixedDelay  线程名称:taskExecutor-4
2020-06-23 23:45:30.005  INFO 34824 : 使用cron  线程名称:taskExecutor-5

2020-06-23 23:45:33.509  INFO 34824 : fixedRate--- 线程名称:taskExecutor-6
2020-06-23 23:45:33.513  INFO 34824 : fixedDelay  线程名称:taskExecutor-7
2020-06-23 23:45:35.005  INFO 34824 : 使用cron  线程名称:taskExecutor-8

...

Cron表达式

cron 表达式是一个字符串,该字符串由 6 个空格分为 7 个域,每一个域代表一个时间含义。 通常定义 “年” 的部分可以省略,实际常用的 Cron 表达式由前 6 部分组成。格式如下:

[秒] [分] [时] [日] [月] [周] [年]

在这里插入图片描述

需要说明的是,Cron 表达式中,“周” 是从周日开始计算的。“周” 域上的 1 表示的是周日,7 表示周六。

Cron中的通配符

, :指的是在两个以上的时间点中都执行。如果在 “分” 这个域中定义为 8,12,35,则表示分别在第8分,第12分,第35分执行该定时任务。

- :指定在某个域的连续范围。如果在 “时” 这个域中定义 1-6,则表示在 1 到 6 点之每小时都触发一次,等价于 1,2,3,4,5,6。

* :表示所有值,可解读为 “每”。 如果在 “日” 这个域中设置 *,表示每一天都会触发。

? :表示不指定值。使用的场景为不需要关心当前设置这个字段的值。例如,要在每月的 8 号触发一个操作,但不关心是周几,我们可以这么设置 0 0 0 8 * ?。

/ :表示触发步进(step),“/” 前面的值代表初始值( “*” 等同 “0”),后面的值代表偏移量,在某个域上周期性触发。比如 在 “秒” 上定义 5/10 表示从 第 5 秒开始,每 10 秒执行一次;而在“分” 上则表示从第 5 分钟开始,每 10 分钟执行一次。

L :表示英文中的 LAST 的意思,只能在 “日” 和 “周” 中使用。在 “日” 中设置,表示当月的最后一天(依据当前月份,如果是二月还会依据是否是润年), 在 “周” 上表示周六,相当于 “7” 或
“SAT”。如果在 “L” 前加上数字,则表示该数据的最后一个。例如在 “周” 上设置 “7L” 这样的格式,则表示 “本月最后一个周六”。

W :表示离指定日期的最近那个工作日(周一至周五)触发,只能在 “日” 中使用且只能用在具体的数字之后。若在 “日” 上置 “15W”,表示离每月 15 号最近的那个工作日触发。假如 15 号正好是周六,则找最近的周五(14号)触发;如果 15 号是周未,则找最近的下周一(16号)触发;如果 15 号正好在工作日(周一至周五),则就在该天触发。如果是 "1W"就只能往本月的下一个最近的工作日推不能跨月往上一个月推。

#: 例如,A#B 表示每月的第 B 个周的周 A(周的计数从周日开始),只能作用于 “周” 上。例如 2#3 表示在每月的第 3 个周二,5#3 表示本月第 3 周的星期四。

注意,L 用在 “周” 这个域上,每周最后一天是周六。“周” 域上的 1 表示的是周日,7 表示周六,即每周计数是从周日开始的。


记录文章地址:https://juejin.cn/post/6844904198752960519

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/647007.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Vue CLI 全局事件总线 消息的订阅与发布

3.10. 全局事件总线(GlobalEventBus) 一种可以在任意组件间通信的方式,本质上就是一个对象,它必须满足以下条件 所有的组件对象都必须能看见他这个对象必须能够使用$on $emit $off方法去绑定、触发和解绑事件 使用步骤 定义全…

MySQL数据库基础 12

第十二章 MySQL数据类型 1. MySQL中的数据类型2. 整数类型2.1 类型介绍2.2 可选属性2.2.1 M2.2.2 UNSIGNED2.2.3 ZEROFILL 2.3 适用场景2.4 如何选择? 3. 浮点类型3.1 类型介绍3.2 数据精度说明3.3 精度误差说明 4. 定点数类型4.1 类型介绍 5. 位类型:BI…

CSS--Java EE

在前端的代码中&#xff0c;CSS 相关的代码写在什么位置呢&#xff1f; CSS 可以写在<style>标签中外部引入&#xff1a;输入 link: css写在 div 标签中 目录 一、选择器的种类 1 基础选择器 1.1 类选择器 1.2 id选择器 1.3 标签选择器 1.4 通用选择器 小结 2 …

Spring Security6 全新写法,大变样!

文章目录 1. WebSecurityConfigurerAdapter2. 使用 Lambda3. 自定义 JSON 登录3.1 自定义 JSON 登录3.1.1 自定义登录过滤器3.1.2 自定义登录接口 3.2 原因分析3.3 问题解决 Spring Security 在最近几个版本中配置的写法都有一些变化&#xff0c;很多常见的方法都废弃了&#x…

Java-IO流基础知识

目录 1.File类与路径知识 1.File类 2.Java中的路径知识 3.创建File类的实例 4.File类的方法使用 5.File类使用的注意点 2.IO流知识 1.IO流原理 2.文件的读入 3.read()的重载方法&#xff1a;难点 4.文件的写出 1.写出的说明 2.写出操作的具体步骤 5.文件的复制&am…

测试工程师如何有效的编写bug报告?

为什么要求有效的缺陷报告&#xff1f; 缺陷报告是测试过程中最重要的部分&#xff0c;对产品的质量有较大的影响&#xff0c;是测试人员价值的终极体现。好的缺陷报告可以减少研发部门的二次缺陷率、提高研发修改缺陷的速度、提高测试部门的信用度、增强测试和研发部门的协作…

Resful API是什么

文章目录 摘要1、RESTful API是什么&#xff1f;2、RESTful是什么&#xff1f;参考资料 摘要 RESTful是整个网络应用程序设计风格和开发方式。而RESTful API是其中API的设计风格。 1、RESTful API是什么&#xff1f; API接口在设计命名时&#xff0c;由版本/操作资源名称/操…

【Spring Boot 初识丨六】依赖注入

上一篇讲了 Spring Boot 的beans 本篇来讲一讲 依赖注入 Spring Boot 初识&#xff1a; 【Spring Boot 初识丨一】入门实战 【Spring Boot 初识丨二】maven 【Spring Boot 初识丨三】starter 【Spring Boot 初识丨四】主应用类 【Spring Boot 初识丨五】beans 依赖注入 一、 定…

攻防世界-Crypto-easychallenge

题目描述&#xff1a;将文件下载下来&#xff0c;只有一个pyc文件 1. 思路分析 先向chatgpt问下什么是pyc文件&#xff1a; OK&#xff0c;这里简单总结下&#xff1a; 1. pyc文件是python源码编译后的生成的二进制文件 2. 通过一些库可以逆向出pyc的源代码 那么我们需要做…

数组题目总结 -- 单调栈问题

目录 零. 单调栈一. Next Greater Element(单调栈问题模板)题目简述&#xff1a;思路和代码&#xff1a;I. 思路II. 代码 二. Next Warmer Weather题目简述&#xff1a;思路和代码&#xff1a;I. 思路II. 代码 三. Next Greater Elements&#xff08;循环数组&#xff09;题目简…

UG\NX二次开发 装配下的点坐标

文章作者:里海 来源网站:https://blog.csdn.net/WangPaiFeiXingYuan 简介: UF_CURVE_create_point 创建一个点 UF_CSYS_create_temp_csys 创建临时坐标系 以上两个函数都有一个点坐标输入,而且都是输入的绝对坐标系下的点坐标值。下面验证在装配下两个绝对坐标值具体指 “当…

【三维视觉】空间点集的最小包围盒计算

0 问题描述 假设有一个空间点集&#xff0c;不重合的点数有N个。 N1时&#xff0c;最小包围盒是一个点&#xff1a;中心为其本身&#xff0c;半径无穷小 N2时&#xff0c;最小包围盒是一个圆&#xff1a;中心为连线中点&#xff0c;半径为边长一半 N3时&#xff0c;不共线的三…

【C#】并行编程实战:任务并行性(上)

在 .NET 的初始版本中&#xff0c;我们只能依赖线程&#xff08;线程可以直接创建或者使用 ThreadPool 类创建&#xff09;。ThreadPool 类提供了一个托管抽象层&#xff0c;但是开发人员仍然需要依靠 Thread 类来进行更好的控制。而 Thread 类维护困难&#xff0c;且不可托管&…

优惠券超发问题该怎么测试?

在拼夕夕面试中&#xff0c;面试官问了一连串经典的问题&#xff1a;“优惠券库存是怎么扣减的&#xff1f;开发为了解决超发优惠券问题而设计的方案&#xff0c;你了解过吗&#xff1f;你又是如何测试的呢&#xff1f;” 当时听到这些问题还挺懵的&#xff0c;没遇到过超发问…

MidJourney教程02

1.主体内容&#xff1a;高数AI你需要画什么&#xff1f;比如说&#xff0c;一个男生在电脑前画画&#xff1f; 2.环境北京&#xff1a;例如给某些地点或者物件&#xff0c;比如桌子上&#xff0c;足球场&#xff0c;水面有倒影等&#xff1f; 3.构图镜头&#xff1a;比如说强…

springboot项目外卖管理 day07-功能补充

文章目录 前端补充功能1、历史订单功能1.1、梳理过程1.2历史订单展示1.3、效果展示 2、修改/删除地址2.1、回显数据梳理过程 代码展示 2.2、修改地址梳理过程代码 2.3、删除地址梳理过程代码展示 3、再来一单功能3.1、梳理过程3.2、具体实现思路&#xff08;参考一下当初我们怎…

Linux操作系统——第四章 进程间通信

目录 进程间通信介绍 进程间通信目的 进程间通信发展 进程间通信分类 管道 System V IPC POSIX IPC 管道 什么是管道 匿名管道 管道读写规则 管道特点 命名管道 创建一个命名管道 匿名管道与命名管道的区别 命名管道的打开规则 system V共享内存 共享内存示意…

【SpringBoot】解决依赖版本不一致报错问题

哈喽大家好&#xff0c;我是阿Q。今天在开发代码的过程中&#xff0c;由于手抖&#xff0c;不知道引入了什么包依赖&#xff0c;导致项目启动一直报错&#xff0c;特写本文来记录下解决问题的经过。 文章目录 问题描述报错信息如下报错描述 解决方法总结 问题描述 报错信息如下…

vite中使用 vite- aliases 插件报错

vite 中使用 vite-aliases 插件报错 vite-aliases 介绍报错内容解决方法 vite-aliases 介绍 vite-aliases 可以帮助我们自动生成别名: 检测你当前目录下包括 src 在内的所有文件夹, 并帮助我们去生成别名。 下载 npm i vite-aliases -D 使用 import { defineConfig } from vi…

VALSE 2023 无锡线下参会个人总结 6月11日-2

VALSE2023无锡线下参会个人总结 6月11日-2 6月11日会议日程安排Workshop&#xff1a;目标检测与分割程明明&#xff1a;粒度自适应的图像感知技术张兆翔&#xff1a;基于多传感器融合的视觉物体检测与分割 Workshop&#xff1a;ChatGPT与计算机视觉白翔&#xff1a;再谈ChatGPT…