实现定时任务几种简单的方式

第一种Timer

Timer 是 JDK 自带的定时任务执行类,无论任何项目都可以直接使用 Timer 来实现定时任务,所以 Timer 的优点就是使用方便。

package com.lsh;

import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.Timer;
import java.util.TimerTask;

/**
 * @author :LiuShihao
 * @date :Created in 2020/8/19 1:35 下午
 * @desc :Timer 是 JDK 自带的定时任务执行类,无论任何项目都可以直接使用 Timer 来实现定时任务,所以 Timer 的优点就是使用方便
 * 缺点:
 * 当一个任务的执行时间过长时,会影响其他任务的调度;
 * 当一个任务抛出异常,其他任务也会终止运行;
 * Timer 类实现定时任务的优点是方便,因为它是 JDK 自定的定时任务,
 * 但缺点是任务如果执行时间太长或者是任务执行异常,会影响其他任务调度,所以在生产环境下建议谨慎使用。
 */
public class MyTimerTask {

    public static void main(String[] args) {
        // 定义一个任务
        TimerTask timerTask = new TimerTask() {
            public void run() {
                System.out.println(LocalDateTime.now().format(DateTimeFormatter.ISO_LOCAL_DATE_TIME));
            }
        };
        // 计时器
        Timer timer = new Timer();
        // 添加执行任务(延迟 1s 执行,每 3s 执行一次)
        timer.schedule(timerTask,1000,10000);
    }
}

缺点:
任务执行时间长影响其他任务
当一个任务的执行时间过长时,会影响其他任务的调度
任务异常影响其他任务
使用 Timer 类实现定时任务时,当一个任务抛出异常,其他任务也会终止运行。

Timer 小结
Timer 类实现定时任务的优点是方便,因为它是 JDK 自定的定时任务,但缺点是任务如果执行时间太长或者是任务执行异常,会影响其他任务调度,所以在生产环境下建议谨慎使用。

第二种ScheduledExecutorService

ScheduledExecutorService 也是 JDK 1.5 自带的 API,我们可以使用它来实现定时任务的功能,也就是说 ScheduledExecutorService 可以实现 Timer 类具备的所有功能,并且它可以解决了 Timer 类存在的所有问题。

package com.lsh;

import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

/**
 * @author :LiuShihao
 * @date :Created in 2020/8/19 1:52 下午
 * @desc :ScheduledExecutorService 也是 JDK 1.5 自带的 API,我们可以使用它来实现定时任务的功能,
 *          也就是说 ScheduledExecutorService 可以实现 Timer 类具备的所有功能,
 *          并且它可以解决了 Timer 类存在的所有问题。
 *
 *          在单机生产环境下建议使用 ScheduledExecutorService 来执行定时任务,
 *          它是 JDK 1.5 之后自带的 API,因此使用起来也比较方便,
 *          并且使用 ScheduledExecutorService 来执行任务,不会造成任务间的相互影响。
 */
public class MyScheduledExecutorService {

    public static void main(String[] args) {
        // 创建任务队列
        // 10 为线程数量
        ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(10);
        // 执行任务
        // 1s 后开始执行,每 3s 执行一次
        scheduledExecutorService.scheduleAtFixedRate(() ->{
            System.out.println(LocalDateTime.now().format(DateTimeFormatter.ISO_LOCAL_DATE_TIME));
        },1,10, TimeUnit.SECONDS);
    }
}

经过测试发现:

  1. 当任务 1 执行时间 5s 超过了执行频率 3s 时,并没有影响任务 2 的正常执行,因此使用 ScheduledExecutorService 可以避免任务执行时间过长对其他任务造成的影响。
  2. 当任务 1 出现异常时,并不会影响任务 2 的执行。

ScheduledExecutorService 小结
在单机生产环境下建议使用 ScheduledExecutorService 来执行定时任务,它是 JDK 1.5 之后自带的 API,因此使用起来也比较方便,并且使用 ScheduledExecutorService 来执行任务,不会造成任务间的相互影响。

第三种Spring框架的定时任务

这种方式应该是我们最常用的方式了。
以 Spring Boot 为例,实现定时任务只需两步:

  1. 开启定时任务;
  2. 添加定时任务;

1.启动类

在主启动类上加上@EnableScheduling注解开启定时任务
在这里插入图片描述

2.定时任务类

在任务类上加上@Component交给spring托管,
在要定时执行的方法上加上@Scheduled(cron = "*/10 * * * * ?")注解即可实现定时执行。代表每10秒执行一次。
corn表达式是控制定时任务执行的时间,可以自行了解。

package com.lsh;

import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;

/**
 * @author :LiuShihao
 * @date :Created in 2020/8/19 2:14 下午
 * @desc :
 */
@Component
public class SpringBootTask {
    private static Boolean isRun = false;

    @Scheduled(cron = "*/5 * * * * ?")
    public void execute() {
        if (isRun){
            return;
        }
        isRun = true;
        System.out.println("定时任务0执行:"+ LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyyMMdd HH:mm:ss")));
        isRun = false;
    }
}

这种是实现定时任务最简单的方式,接下来我们学习一下如何动态的实现定时任务SpringBoot实现动态的定时任务

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Liu_Shihao

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值