java定时任务_定时任务3种实现方式

java定时任务_定时任务3种实现方式

  • 定时任务在实际的开发中特别常见,比如电商平台 30 分钟后自动取消未支付的订单,以及凌晨的数据汇总和备份等,都需要借助定时任务来实现,那么我们本文就来看一下定时任务最简单的几种实现方式。

Timer

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

    public class MyTimerTask {
        public static void main(String[] args) {
            // 定义一个任务
            TimerTask timerTask = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("Run timerTask:" + new Date());
                }
            };
            // 计时器
            Timer timer = new Timer();
            // 添加执行任务(延迟 1s 执行,每 3s 执行一次)
            timer.schedule(timerTask, 1000, 3000);
        }
    }
    
  • 程序执行结果如下:

  • Run timerTask:Mon Aug 17 21:29:25 CST 2020
    Run timerTask:Mon Aug 17 21:29:28 CST 2020
    Run timerTask:Mon Aug 17 21:29:31 CST 2020
    

Timer 缺点分析

  • 任务执行时间长影响其他任务

  • 当一个任务的执行时间过长时,会影响其他任务的调度,如下代码所示:

    public class MyTimerTask {
        public static void main(String[] args) {
            // 定义任务 1
            TimerTask timerTask = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("进入 timerTask 1:" + new Date());
                    try {
                        // 休眠 5 秒
                        TimeUnit.SECONDS.sleep(5);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println("Run timerTask 1:" + new Date());
                }
            };
            // 定义任务 2
            TimerTask timerTask2 = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("Run timerTask 2:" + new Date());
                }
            };
            // 计时器
            Timer timer = new Timer();
            // 添加执行任务(延迟 1s 执行,每 3s 执行一次)
            timer.schedule(timerTask, 1000, 3000);
            timer.schedule(timerTask2, 1000, 3000);
        }
    }
    
  • 程序执行结果如下:

  • 进入 timerTask 1Mon Aug 17 21:44:08 CST 2020
    Run timerTask 1Mon Aug 17 21:44:13 CST 2020 Run timerTask 2Mon Aug 17 21:44:13 CST 2020
    进入 timerTask 1Mon Aug 17 21:44:13 CST 2020
    Run timerTask 1Mon Aug 17 21:44:18 CST 2020
    进入 timerTask 1Mon Aug 17 21:44:18 CST 2020
    Run timerTask 1Mon Aug 17 21:44:23 CST 2020 Run timerTask 2Mon Aug 17 21:44:23 CST 2020
    进入 timerTask 1Mon Aug 17 21:44:23 CST 2020
    
  • 从上述结果中可以看出,当任务 1 运行时间超过设定的间隔时间时,任务 2 也会延迟执行。 原本任务 1 和任务 2 的执行时间间隔都是 3s,但因为任务 1 执行了 5s,因此任务 2 的执行时间间隔也变成了 10s(和原定时间不符)

  • 任务异常影响其他任务

  • 使用 Timer 类实现定时任务时,当一个任务抛出异常,其他任务也会终止运行,如下代码所示:

  • public class MyTimerTask {
        public static void main(String[] args) {
            // 定义任务 1
            TimerTask timerTask = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("进入 timerTask 1:" + new Date());
                    // 模拟异常
                    int num = 8 / 0;
                    System.out.println("Run timerTask 1:" + new Date());
                }
            };
            // 定义任务 2
            TimerTask timerTask2 = new TimerTask() {
                @Override
                public void run() {
                    System.out.println("Run timerTask 2:" + new Date());
                }
            };
            // 计时器
            Timer timer = new Timer();
            // 添加执行任务(延迟 1s 执行,每 3s 执行一次)
            timer.schedule(timerTask, 1000, 3000);
            timer.schedule(timerTask2, 1000, 3000);
        }
    }
    
  • 程序执行结果如下:

  • 进入 timerTask 1Mon Aug 17 22:02:37 CST 2020
    Exception in thread "Timer-0" java.lang.ArithmeticException: / by zero
    at com.example.MyTimerTask$1.run(MyTimerTask.java:21)
    at java.util.TimerThread.mainLoop(Timer.java:555)
    at java.util.TimerThread.run(Timer.java:505)
    Process finished with exit code 0
    

Timer 小结

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

ScheduledExecutorService

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

  • ScheduledExecutorService 实现定时任务的代码示例如下:

  • public class MyScheduledExecutorService {
        public static void main(String[] args) {
            // 创建任务队列
            ScheduledExecutorService scheduledExecutorService =
                    Executors.newScheduledThreadPool(10); // 10 为线程数量
      // 执行任务
            scheduledExecutorService.scheduleAtFixedRate(() -> {
                System.out.println("Run Schedule:" + new Date());
            }, 1, 3, TimeUnit.SECONDS); // 1s 后开始执行,每 3s 执行一次
        }
    }
    
  • 程序执行结果如下:

  • Run ScheduleMon Aug 17 21:44:23 CST 2020
    Run ScheduleMon Aug 17 21:44:26 CST 2020
    Run ScheduleMon Aug 17 21:44:29 CST 2020
    
  • 使用 ScheduledExecutorService 可以避免任务执行时间过长对其他任务造成的影响

  • 当任务 1 出现异常时,并不会影响任务 2 的执行

ScheduledExecutorService 小结

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

Spring Task

  • 如果使用的是 Spring 或 Spring Boot 框架,可以直接使用 Spring Framework 自带的定时任务,使用上面两种定时任务的实现方式,很难实现设定了具体时间的定时任务,比如当我们需要每周五来执行某项任务时,但如果使用 Spring Task 就可轻松的实现此需求
  • 以 Spring Boot 为例,实现定时任务只需两步:
  • 开启定时任务
  • 添加定时任务

开启定时任务

  • 开启定时任务只需要在 Spring Boot 的启动类上声明 @EnableScheduling 即可,实现代码如下:

  • @SpringBootApplication
    @EnableScheduling // 开启定时任务
    public class DemoApplication {
        // do someing
    }
    

添加定时任务

  • 定时任务的添加只需要使用 @Scheduled 注解标注即可,如果有多个定时任务可以创建多个 @Scheduled 注解标注的方法,示例代码如下:

  • import org.springframework.scheduling.annotation.Scheduled;
    import org.springframework.stereotype.Component;
     
    @Component // 把此类托管给 Spring,不能省略
    public class TaskUtils {
        // 添加定时任务
        @Scheduled(cron = "59 59 23 0 0 5") // cron 表达式,每周五 23:59:59 执行
        public void doTask(){
            System.out.println("我是定时任务~");
        }
    }
    
  • 定时任务是自动触发的无需手动干预,也就是说 Spring Boot 启动后会自动加载并执行定时任务。

  • 17
    点赞
  • 57
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 8
    评论
Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源项目,它可以与J2EE与J2SE应用程序相结合也可以单独使用。Quartz可以用来创建简单或为运行十个,百个,甚至是好几万个Jobs这样复杂的程序。Jobs可以做成标准的Java组件或 EJBs。 Quartz的优势: 1、Quartz是一个任务调度框架(库),它几乎可以集成到任何应用系统中。 2、Quartz是非常灵活的,它让您能够以最“自然”的方式来编写您的项目的代码,实现您所期望的行为 3、Quartz是非常轻量级的,只需要非常少的配置 —— 它实际上可以被跳出框架来使用,如果你的需求是一些相对基本的简单的需求的话。 4、Quartz具有容错机制,并且可以在重启服务的时候持久化(”记忆”)你的定时任务,你的任务也不会丢失。 5、可以通过Quartz,封装成自己的分布式任务调度,实现强大的功能,成为自己的产品。6、有很多的互联网公司也都在使用Quartz。比如美团 Spring是一个很优秀的框架,它无缝的集成了Quartz,简单方便的让企业级应用更好的使用Quartz进行任务的调度。   课程说明:在我们的日常开发中,各大型系统的开发少不了任务调度,简单的单机任务调度已经满足不了我们的系统需求,复杂的任务会让程序猿头疼, 所以急需一套专门的框架帮助我们去管理定时任务,并且可以在多台机器去执行我们的任务,还要可以管理我们的分布式定时任务。本课程从Quartz框架讲起,由浅到深,从使用到结构分析,再到源码分析,深入解析Quartz、Spring+Quartz,并且会讲解相关原理, 让大家充分的理解这个框架和框架的设计思想。由于互联网的复杂性,为了满足我们特定的需求,需要对Spring+Quartz进行二次开发,整个二次开发过程都会进行讲解。Spring被用在了越来越多的项目中, Quartz也被公认为是比较好用的定时器设置工具,学完这个课程后,不仅仅可以熟练掌握分布式定时任务,还可以深入理解大型框架的设计思想。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

IT枫斗者

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

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

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

打赏作者

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

抵扣说明:

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

余额充值