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

前言

定时任务通常有三种完成方法

  1. java自带的API java.util.Timer类 java.util.TimerTask类

在 JDK 中,内置了两个类,可以实现定时任务的功能:

  • java.util.Timer :可以通过创建 java.util.TimerTask 调度任务,在同一个线程中串行执行,相互影响。也就是说,对于同一个 Timer 里的多个 TimerTask 任务,如果一个 TimerTask 任务在执行中,其它 TimerTask 即使到达执行的时间,也只能排队等待。因为 Timer 是串行的,同时存在 坑坑 ,所以后来 JDK 又推出了 ScheduledExecutorService ,Timer 也基本不再使用。

  • java.util.concurrent.ScheduledExecutorService :在 JDK 1.5 新增,基于线程池设计的定时任务类,每个调度任务都会被分配到线程池中并发执行,互不影响。这样,ScheduledExecutorService 就解决了 Timer 串行的问题。

在日常开发中,我们很少直接使用 Timer 或 ScheduledExecutorService 来实现定时任务的需求。主要有几点原因:

  • 它们仅支持按照指定频率,不直接支持指定时间的定时调度,需要我们结合 Calendar 自行计算,才能实现复杂时间的调度。例如说,每天、每周五、2019-11-11 等等。

  • 它们是进程级别,而我们为了实现定时任务的高可用,需要部署多个进程。此时需要等多考虑,多个进程下,同一个任务在相同时刻,不能重复执行。

  • 项目可能存在定时任务较多,需要统一的管理,此时不得不进行二次封装。

     所以,一般情况下,我们会选择专业的调度任务中间件

2、Quartz框架 开源 功能强大 使用起来稍显复杂

3.Spring 3.0以后自带了task 调度工具,比Quartz更加的简单方便

使用SpringBoot创建定时任务非常简单,目前主要有以下三种创建方式:
一、基于注解(@Scheduled)
二、基于接口(SchedulingConfigurer) 前者相信大家都很熟悉,但是实际使用中我们往往想从数据库中读取指定时间来动态执行定时任务,这时候基于接口的定时任务就派上用场了。
三、基于注解设定多线程定时任务
 

 

一、静态:基于注解

1、创建定时器

使用SpringBoot基于注解来创建定时任务非常简单,只需几行代码便可完成。 代码如下:

 

@Component
@Configuration      //1.主要用于标记配置类,兼备Component的效果。
@EnableScheduling   // 2.开启定时任务
public class SaticScheduleTask {
    //3.添加定时任务
    @Scheduled(cron = "0/5 * * * * ?")
    //或直接指定时间间隔,例如:5秒
    //@Scheduled(fixedRate=5000)
    private void configureTasks() {
        System.err.println("执行静态定时任务时间: " + LocalDateTime.now());
    }
}

@Controller
public class TestSchudle {


    @Scheduled(cron = "0/5 * * * * ?")
    //或直接指定时间间隔,例如:5秒
    //@Scheduled(fixedRate=5000)
    private void configureTasks() {
        System.err.println("在controller里进行定时操作 " + LocalDateTime.now());
    }

}

关于Cron表达式介绍

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

 

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

关于Cron表达式的介绍来源:

http://blog.youkuaiyun.com/supingemail/article/details/22274279

表达式网站生成:

表达式网站地址

 

注意:

Spring允许的表达式是所有可能的cron表达式的子集,例如它不包含year字段,并且不能使用所有特殊字符。如:

//每个月的最后一天的23点59分

@Scheduled(cron = "0 59 23 L * ?")

这个表达式错误,因为 有特殊字符L(W也是不能用的),代码运行提示

 Encountered invalid @Scheduled method 'deleteBatchByDate': For input string: "L"

 

二、动态:基于接口

基于接口(SchedulingConfigurer)

 

<parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter</artifactId>
        <version>2.0.4.RELEASE</version>
    </parent>

    <dependencies>
        <dependency><!--添加Web依赖 -->
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency><!--添加MySql依赖 -->
             <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
        </dependency>
        <dependency><!--添加Mybatis依赖 配置mybatis的一些初始化的东西-->
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>1.3.1</version>
        </dependency>
        <dependency><!-- 添加mybatis依赖 -->
            <groupId>org.mybatis</groupId>
            <artifactId>mybatis</artifactId>
            <version>3.4.5</version>
            <scope>compile</scope>
        </dependency>
    </dependencies>

 

2、添加数据库记录:

开启本地数据库mysql,随便打开查询窗口,然后执行脚本内容,如下:

 

DROP DATABASE IF EXISTS `socks`;
CREATE DATABASE `socks`;
USE `SOCKS`;
DROP TABLE IF EXISTS `cron`;
CREATE TABLE `cron`  (
  `cron_id` varchar(30) NOT NULL PRIMARY KEY,
  `cron` varchar(30) NOT NULL  
);
INSERT INTO `cron` VALUES ('1', '0/5 * * * * ?');

 

 

 

然后在项目中的application.yml 添加数据源:

spring:
  datasource:
    url: jdbc:mysql://localhost:3306/socks
    username: root
    password: 123456

3、创建定时器

数据库准备好数据之后,我们编写定时任务,注意这里添加的是TriggerTask,目的是循环读取我们在数据库设置好的执行周期,以及执行相关定时任务的内容。
具体代码如下:

 

@Component
@Configuration      //1.主要用于标记配置类,兼备Component的效果。
@EnableScheduling   // 2.开启定时任务
public class DynamicScheduleTask implements SchedulingConfigurer {

    @Mapper
    public interface CronMapper {
        @Select("select cron from cron limit 1")
        public String getCron();
    }

    @Autowired      //注入mapper
    @SuppressWarnings("all")
    CronMapper cronMapper;

    /**
     * 执行定时任务.
     */
    @Override
    public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {

        taskRegistrar.addTriggerTask(
                //1.添加任务内容(Runnable)
                () -> System.out.println("执行动态定时任务: " + LocalDateTime.now().toLocalTime()),
                //2.设置执行周期(Trigger)
                triggerContext -> {
                    //2.1 从数据库获取执行周期
                    String cron = cronMapper.getCron();
                    //2.2 合法性校验.
                    if (StringUtils.isEmpty(cron)) {
                        // Omitted Code ..
                    }
                    //2.3 返回执行周期(Date)
                    return new CronTrigger(cron).nextExecutionTime(triggerContext);
                }
        );
    }

}

三、多线程定时任务

基于注解设定多线程定时任务

1、创建多线程定时任务

 

//@Component注解用于对那些比较中立的类进行注释;
//相对与在持久层、业务层和控制层分别采用 @Repository、@Service 和 @Controller 对分层中的类进行注释
@Component
@EnableScheduling   // 1.开启定时任务
@EnableAsync        // 2.开启多线程
public class MultithreadScheduleTask {

        @Async
        @Scheduled(fixedDelay = 1000)  //间隔1秒
        public void first() throws InterruptedException {
            System.out.println("第一个定时任务开始 : " + LocalDateTime.now().toLocalTime() + "\r\n线程 : " + Thread.currentThread().getName());
            System.out.println();
            Thread.sleep(1000 * 10);
        }

        @Async
        @Scheduled(fixedDelay = 2000)
        public void second() {
            System.out.println("第二个定时任务开始 : " + LocalDateTime.now().toLocalTime() + "\r\n线程 : " + Thread.currentThread().getName());
            System.out.println();
        }
    }

 

实例

参考:https://blog.youkuaiyun.com/debugbugbg/article/details/81091715

 

              SpringTask 默认是单线程的 上面的两个定时任务使用的都是同一个线程;在实际开发中,不希望所有的任务都运行在一个线程中,想要改成多线程,给SpringTask提供一个多线程的TaskScheduler,Spring已经有默认实现

结果:

 

 

代码地址:https://github.com/mmzsblog/springboot-schedule

 转载:https://www.cnblogs.com/mmzs/p/10161936.html

其他参考:

https://blog.youkuaiyun.com/jack_bob/article/details/78786740

https://blog.youkuaiyun.com/fastlearn/article/details/83306796

https://mp.weixin.qq.com/s/jqN4noo5NazckPCehWFgpA

在 Spring Boot 中使用 `@Scheduled` 注解实现定时任务是一种简单且高效的方式,能够通过注解直接定义方法的执行周期。以下是具体的实现方式: ### 1. 启用定时任务支持 要在 Spring Boot 应用中启用定时任务功能,需要在主类或配置类上添加 `@EnableScheduling` 注解。此注解会激活对 `@Scheduled` 注解支持,并确保 Spring 容器调度相关的任务。 例如,在主类上启用定时任务: ```java @SpringBootApplication @EnableScheduling public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } } ``` 这种方式适用于大多数 Spring Boot 应用程序[^2]。 如果使用的是传统的 Spring 应用程序,则可以通过配置类启用定时任务: ```java @Configuration @EnableScheduling public class ScheduleConfig { // 配置内容 } ``` 这种方式用于显式地开启对 `@Scheduled` 的支持[^3]。 --- ### 2. 创建定时任务 创建一个由 Spring 管理的 Bean 类,并在类中定义需要定时执行的方法。该类需要使用 `@Component` 注解标记为组件,以便被 Spring 容器管理。在目标方法上使用 `@Scheduled` 注解来指定执行周期。 例如,定义一个每 5 秒执行一次的任务: ```java import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component; @Component public class SimpleScheduledTask { @Scheduled(fixedRate = 5000) // 每隔5秒执行一次任务 public void executeTask() { System.out.println("定时任务执行了"); } } ``` 上述代码中,`fixedRate = 5000` 表示方法每隔 5000 毫秒(即 5 秒)执行一次[^2]。 --- ### 3. 使用 cron 表达式配置复杂时间周期 除了简单的固定间隔外,`@Scheduled` 注解支持使用 **cron 表达式** 来定义更复杂的时间周期。cron 表达式允许精确控制任务的执行时间,包括秒、分、小时、日、月和星期等字段。 例如,以下代码表示每天凌晨 1:00 执行任务: ```java @Scheduled(cron = "0 0 1 * * ?") public void dailyTask() { System.out.println("每日凌晨1点执行任务"); } ``` cron 表达式的格式为:`秒 分 时 日 月 星期`,每个字段分别代表不同的时间单位,具体规则可以参考在线生成工具 [cron表达式在线生成器](http://cron.ciding.cc/)[^4]。 --- ### 4. 其他常用参数 除了 `fixedRate` 和 `cron` 外,`@Scheduled` 注解支持其他参数: - `fixedDelay`: 上次任务执行完成后,延迟一定时间再次执行。 - `initialDelay`: 首次执行前延迟一定时间。 例如: ```java @Scheduled(fixedDelay = 3000, initialDelay = 1000) public void delayedTask() { System.out.println("首次延迟1秒后执行,之后每次延迟3秒"); } ``` --- ### 总结 在 Spring Boot 中实现定时任务的关键步骤是: 1. 在主类或配置类上添加 `@EnableScheduling` 注解以启用定时任务支持。 2. 创建一个由 Spring 管理的 Bean 类,并在类中定义方法。 3. 在方法上使用 `@Scheduled` 注解,并根据需求选择合适的参数(如 `fixedRate`, `cron`, `fixedDelay` 等)来定义执行周期。 这种方式不仅简洁明了,而且易于维护和扩展,适合大多数基于 Spring Boot 的应用场景。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值