定時任務最簡單的3種實現方法(超好用)

這是個人第 86 篇原創文章
java

做者 | 王磊web

來源 | Java中文社羣(ID:javacn666)面試

轉載請聯繫受權(微信ID:GG_Stone)

定時任務在實際的開發中特別常見,好比電商平臺 30 分鐘後自動取消未支付的訂單,以及凌晨的數據彙總和備份等,都須要藉助定時任務來實現,那麼咱們本文就來看一下定時任務最簡單的幾種實現方式。
redis

TOP 1:Timer

Timer 是 JDK 自帶的定時任務執行類,不管任何項目均可以直接使用 Timer 來實現定時任務,因此 Timer 的優勢就是使用方便,它的實現代碼以下:spring

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, 10003000);
    }
}

程序執行結果以下:微信

Run timerTask:Mon Aug 17 21:29:25 CST 2020app

Run timerTask:Mon Aug 17 21:29:28 CST 2020框架

Run timerTask:Mon Aug 17 21:29:31 CST 2020編輯器

Timer 缺點分析

Timer 類實現定時任務雖然方便,但在使用時須要注意如下問題。分佈式

問題 1:任務執行時間長影響其餘任務

當一個任務的執行時間過長時,會影響其餘任務的調度,以下代碼所示:

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, 10003000);
        timer.schedule(timerTask2, 10003000);
    }
}

程序執行結果以下:

進入 timerTask 1:Mon Aug 17 21:44:08 CST 2020

Run timerTask 1:Mon Aug 17 21:44:13 CST 2020

Run timerTask 2:Mon Aug 17 21:44:13 CST 2020

進入 timerTask 1:Mon Aug 17 21:44:13 CST 2020

Run timerTask 1:Mon Aug 17 21:44:18 CST 2020

進入 timerTask 1:Mon Aug 17 21:44:18 CST 2020

Run timerTask 1:Mon Aug 17 21:44:23 CST 2020

Run timerTask 2:Mon Aug 17 21:44:23 CST 2020

進入 timerTask 1:Mon Aug 17 21:44:23 CST 2020

從上述結果中能夠看出,當任務 1 運行時間超過設定的間隔時間時,任務 2 也會延遲執行。 本來任務 1 和任務 2 的執行時間間隔都是 3s,但由於任務 1 執行了 5s,所以任務 2 的執行時間間隔也變成了 10s(和原定時間不符)。

問題 2:任務異常影響其餘任務

使用 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, 10003000);
        timer.schedule(timerTask2, 10003000);
    }
}

程序執行結果以下:

進入 timerTask 1:Mon 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 自定的定時任務,但缺點是任務若是執行時間太長或者是任務執行異常,會影響其餘任務調度,因此在生產環境下建議謹慎使用。

TOP 2: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());
        }, 13, TimeUnit.SECONDS); // 1s 後開始執行,每 3s 執行一次
    }
}

程序執行結果以下:

Run Schedule:Mon Aug 17 21:44:23 CST 2020

Run Schedule:Mon Aug 17 21:44:26 CST 2020

Run Schedule:Mon Aug 17 21:44:29 CST 2020

ScheduledExecutorService 可靠性測試

① 任務超時執行測試

ScheduledExecutorService 能夠解決 Timer 任務之間相應影響的缺點,首先咱們來測試一個任務執行時間過長,會不會對其餘任務形成影響,測試代碼以下:

public class MyScheduledExecutorService {
    public static void main(String[] args) {
        // 建立任務隊列
        ScheduledExecutorService scheduledExecutorService =
                Executors.newScheduledThreadPool(10);
        // 執行任務 1
        scheduledExecutorService.scheduleAtFixedRate(() -> {
            System.out.println("進入 Schedule:" + new Date());
            try {
                // 休眠 5 秒
                TimeUnit.SECONDS.sleep(5);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("Run Schedule:" + new Date());
        }, 13, TimeUnit.SECONDS); // 1s 後開始執行,每 3s 執行一次
        // 執行任務 2
        scheduledExecutorService.scheduleAtFixedRate(() -> {
            System.out.println("Run Schedule2:" + new Date());
        }, 13, TimeUnit.SECONDS); // 1s 後開始執行,每 3s 執行一次
    }
}

程序執行結果以下:

Run Schedule2:Mon Aug 17 11:27:55 CST 2020

進入 Schedule:Mon Aug 17 11:27:55 CST 2020

Run Schedule2:Mon Aug 17 11:27:58 CST 2020

Run Schedule:Mon Aug 17 11:28:00 CST 2020

進入 Schedule:Mon Aug 17 11:28:00 CST 2020

Run Schedule2:Mon Aug 17 11:28:01 CST 2020

Run Schedule2:Mon Aug 17 11:28:04 CST 2020

從上述結果能夠看出,當任務 1 執行時間 5s 超過了執行頻率 3s 時,並無影響任務 2 的正常執行,所以使用 ScheduledExecutorService 能夠避免任務執行時間過長對其餘任務形成的影響

② 任務異常測試

接下來咱們來測試一下 ScheduledExecutorService 在一個任務異常時,是否會對其餘任務形成影響,測試代碼以下:

public class MyScheduledExecutorService {
    public static void main(String[] args) {
        // 建立任務隊列
        ScheduledExecutorService scheduledExecutorService =
                Executors.newScheduledThreadPool(10);
        // 執行任務 1
        scheduledExecutorService.scheduleAtFixedRate(() -> {
            System.out.println("進入 Schedule:" + new Date());
            // 模擬異常
            int num = 8 / 0;
            System.out.println("Run Schedule:" + new Date());
        }, 13, TimeUnit.SECONDS); // 1s 後開始執行,每 3s 執行一次
        // 執行任務 2
        scheduledExecutorService.scheduleAtFixedRate(() -> {
            System.out.println("Run Schedule2:" + new Date());
        }, 13, TimeUnit.SECONDS); // 1s 後開始執行,每 3s 執行一次
    }
}

程序執行結果以下:

進入 Schedule:Mon Aug 17 22:17:37 CST 2020

Run Schedule2:Mon Aug 17 22:17:37 CST 2020

Run Schedule2:Mon Aug 17 22:17:40 CST 2020

Run Schedule2:Mon Aug 17 22:17:43 CST 2020

從上述結果能夠看出,當任務 1 出現異常時,並不會影響任務 2 的執行

ScheduledExecutorService 小結

在單機生產環境下建議使用 ScheduledExecutorService 來執行定時任務,它是 JDK 1.5 以後自帶的 API,所以使用起來也比較方便,而且使用 ScheduledExecutorService 來執行任務,不會形成任務間的相互影響。

TOP 3:Spring Task

若是使用的是 Spring 或 Spring Boot 框架,能夠直接使用 Spring Framework 自帶的定時任務,使用上面兩種定時任務的實現方式,很難實現設定了具體時間的定時任務,好比當咱們須要每週五來執行某項任務時,但若是使用 Spring Task 就可輕鬆的實現此需求。

以 Spring Boot 爲例,實現定時任務只需兩步:

  1. 開啓定時任務;
  2. 添加定時任務。

具體實現步驟以下。

① 開啓定時任務

開啓定時任務只須要在 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 啓動後會自動加載並執行定時任務。

Cron 表達式

Spring Task 的實現須要使用 cron 表達式來聲明執行的頻率和規則,cron 表達式是由 6 位或者 7 位組成的(最後一位能夠省略),每位之間以空格分隔,每位從左到右表明的含義以下:

其中 * 和 ? 號都表示匹配全部的時間。

cron 表達式在線生成地址:https://cron.qqe2.com/

知識擴展:分佈式定時任務

上面的方法都是關於單機定時任務的實現,若是是分佈式環境可使用 Redis 來實現定時任務。

使用 Redis 實現延遲任務的方法大致可分爲兩類:經過 ZSet 的方式和鍵空間通知的方式

① ZSet 實現方式

經過 ZSet 實現定時任務的思路是,將定時任務存放到 ZSet 集合中,而且將過時時間存儲到 ZSet 的 Score 字段中,而後經過一個無線循環來判斷當前時間內是否有須要執行的定時任務,若是有則進行執行,具體實現代碼以下:

import redis.clients.jedis.Jedis;
import utils.JedisUtils;
import java.time.Instant;
import java.util.Set;

public class DelayQueueExample {
    // zset key
    private static final String _KEY = "myTaskQueue";
    
    public static void main(String[] args) throws InterruptedException {
        Jedis jedis = JedisUtils.getJedis();
        // 30s 後執行
        long delayTime = Instant.now().plusSeconds(30).getEpochSecond();
        jedis.zadd(_KEY, delayTime, "order_1");
        // 繼續添加測試數據
        jedis.zadd(_KEY, Instant.now().plusSeconds(2).getEpochSecond(), "order_2");
        jedis.zadd(_KEY, Instant.now().plusSeconds(2).getEpochSecond(), "order_3");
        jedis.zadd(_KEY, Instant.now().plusSeconds(7).getEpochSecond(), "order_4");
        jedis.zadd(_KEY, Instant.now().plusSeconds(10).getEpochSecond(), "order_5");
        // 開啓定時任務隊列
        doDelayQueue(jedis);
    }

    /**
     * 定時任務隊列消費
     * @param jedis Redis 客戶端
     */

    public static void doDelayQueue(Jedis jedis) throws InterruptedException {
        while (true) {
            // 當前時間
            Instant nowInstant = Instant.now();
            long lastSecond = nowInstant.plusSeconds(-1).getEpochSecond(); // 上一秒時間
            long nowSecond = nowInstant.getEpochSecond();
            // 查詢當前時間的全部任務
            Set<String> data = jedis.zrangeByScore(_KEY, lastSecond, nowSecond);
            for (String item : data) {
                // 消費任務
                System.out.println("消費:" + item);
            }
            // 刪除已經執行的任務
            jedis.zremrangeByScore(_KEY, lastSecond, nowSecond);
            Thread.sleep(1000); // 每秒查詢一次
        }
    }
}

② 鍵空間通知

咱們能夠經過 Redis 的鍵空間通知來實現定時任務,它的實現思路是給全部的定時任務設置一個過時時間,等到了過時以後,咱們經過訂閱過時消息就能感知到定時任務須要被執行了,此時咱們執行定時任務便可。

默認狀況下 Redis 是不開啓鍵空間通知的,須要咱們經過 config set notify-keyspace-events Ex 的命令手動開啓,開啓以後定時任務的代碼以下:

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPubSub;
import utils.JedisUtils;

public class TaskExample {
    public static final String _TOPIC = "__keyevent@0__:expired"// 訂閱頻道名稱
    public static void main(String[] args) {
        Jedis jedis = JedisUtils.getJedis();
        // 執行定時任務
        doTask(jedis);
    }

    /**
     * 訂閱過時消息,執行定時任務
     * @param jedis Redis 客戶端
     */

    public static void doTask(Jedis jedis) {
        // 訂閱過時消息
        jedis.psubscribe(new JedisPubSub() {
            @Override
            public void onPMessage(String pattern, String channel, String message) {
                // 接收到消息,執行定時任務
                System.out.println("收到消息:" + message);
            }
        }, _TOPIC);
    }
}

更多關於定時任務的實現,請點擊《史上最全的延遲任務實現方式彙總!附代碼》。

   


往期推薦

史上最全的延遲任務實現方式彙總!附代碼(強烈推薦)


磊哥最近面試了好多人,聊聊個人感覺!(附面試知識點)


關注下方二維碼,查看更多幹貨!


本文分享自微信公衆號 - Java中文社羣(javacn666)。
若有侵權,請聯繫 support@oschina.cn 刪除。
本文參與「OSC源創計劃」,歡迎正在閱讀的你也加入,一塊兒分享。

相關文章
相關標籤/搜索