Java 定时任务调度(2)--Timer 及 ScheduledThreadPoolExecutor 介绍

本文主要介绍使用 Java 原生的 java.util.Timer 和 java.util.concurrent.ScheduledThreadPoolExecutor 来实现定时任务;本文中所使用到的 Java 版本:jdk1.8.0_181。

1、java.util.Timer 使用

@Test
public void test1() throws InterruptedException {
    Timer timer = new Timer();

    timer.schedule(new TimerTask() {
        @Override
        public void run() {
            logger.info("在一个时间点运行一次");
            throw new RuntimeException();
        }
    }, new Date(System.currentTimeMillis() + 2000));

    timer.schedule(new TimerTask() {
        @Override
        public void run() {
            logger.info("延迟3s运行一次");
        }
    }, 3000);

    timer.schedule(new TimerTask() {
        @Override
        public void run() {
            logger.info("延迟4s运行第一次,然后每隔2s运行一次");
        }
    }, 4000, 2000);

    timer.schedule(new TimerTask() {
        @Override
        public void run() {
            logger.info("在一个时间点运行第一次,然后每隔2s运行一次");
        }
    }, new Date(System.currentTimeMillis() + 1000 * 5), 2000);

    Thread.sleep(1000 * 20);
}

@Test
public void test2() throws InterruptedException {
    Timer timer = new Timer();
    timer.scheduleAtFixedRate(new TimerTask() {
        @Override
        public void run() {
            logger.info("延迟3s运行第一次,然后每隔2s运行一次");
        }
    }, 3000, 2000);

    timer.scheduleAtFixedRate(new TimerTask() {
        @Override
        public void run() {
            logger.info("在一个时间点运行第一次,然后每隔2s运行一次");
        }
    }, new Date(System.currentTimeMillis() + 4000), 2000);

    Thread.sleep(1000 * 20);
}

java.util.Timer 有如下缺点:
1、java.util.Timer 是单线程执行任务,如果多个任务在一个时间点执行,会造成任务不能够在准确的时间点执行
2、java.util.Timer 执行某个任务过程中如果抛出了异常,那么执行线程将会终止,导致其他的任务也不能再执行
基于 java.util.Timer 的缺点,可使用 java.util.concurrent.ScheduledThreadPoolExecutor 来代替 java.util.Timer,ScheduledThreadPoolExecutor 使用线程池来执行任务,不存在上面的缺点。

2、java.util.concurrent.ScheduledThreadPoolExecutor 使用

@Test
public void test3() throws InterruptedException {
    ScheduledThreadPoolExecutor executor = new ScheduledThreadPoolExecutor(1);
    executor.schedule(() -> {
        logger.info("延迟2s运行一次");
    }, 2, TimeUnit.SECONDS);

    executor.schedule(() -> {
        logger.info("延迟3s运行一次");
        return null;
    }, 3, TimeUnit.SECONDS);

    executor.scheduleAtFixedRate(() -> {
        logger.info("延迟4s运行第一次,然后每隔2s运行一次");
    }, 4, 2, TimeUnit.SECONDS);

    executor.scheduleWithFixedDelay(() -> {
        logger.info("延迟5s运行第一次,上一次任务运行结束后等2s再运行下一次任务。");
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }, 5, 2, TimeUnit.SECONDS);

    Thread.sleep(1000 * 20);
}

 

posted @ 2021-05-08 16:37  且行且码  阅读(278)  评论(0编辑  收藏  举报