欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

使用ScheduledThreadPoolExecutor踩過最痛的坑

 更新時間:2023年08月10日 16:57:39   作者:肥肥技術(shù)宅  
這篇文章主要介紹了使用ScheduledThreadPoolExecutor踩過最痛的坑及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教

概述

最近項目上反饋某個重要的定時任務(wù)突然不執(zhí)行了,很頭疼,開發(fā)環(huán)境和測試環(huán)境都沒有出現(xiàn)過這個問題。

定時任務(wù)采用的是 ScheduledThreadPoolExecutor ,后來一看代碼發(fā)現(xiàn)踩了一個大坑....

還原"大坑"

這個坑就是如果 ScheduledThreadPoolExecutor 中執(zhí)行的任務(wù)出錯拋出異常后,不僅不會打印異常堆棧信息,同時還會取消后面的調(diào)度 

直接看例子

@Test
public void testException() throws InterruptedException {
    // 創(chuàng)建1個線程的調(diào)度任務(wù)線程池
    ScheduledExecutorService scheduledExecutorService = Executors.newSingleThreadScheduledExecutor();
    // 創(chuàng)建一個任務(wù)
    Runnable runnable = new Runnable() {
        volatile int num = 0;
        @Override
        public void run() {
            num ++;
            // 模擬執(zhí)行報錯
            if(num > 5) {
                throw new RuntimeException("執(zhí)行錯誤");
            }
            log.info("exec num: [{}].....", num);
        }
    };
    // 每隔1秒鐘執(zhí)行一次任務(wù)
    scheduledExecutorService.scheduleAtFixedRate(runnable, 0, 1, TimeUnit.SECONDS);
    Thread.sleep(10000);
}

運(yùn)行結(jié)果:

  • 只執(zhí)行了5次后,就不打印,不執(zhí)行了,因為報錯了
  • 任務(wù)報錯,也沒有打印一次堆棧,更導(dǎo)致調(diào)度任務(wù)取消,后果十分嚴(yán)重。

解決方案

解決方法也非常簡單,只要通過try catch捕獲異常即可。

運(yùn)行結(jié)果:

看到不僅打印了異常堆棧,而且也會進(jìn)行周期性的調(diào)度。

更推薦的做法

更好的建議可以在自己的項目中封裝一個包裝類,要求所有的調(diào)度都提交通過我們統(tǒng)一的包裝類, 如下代碼:

@Slf4j
public class RunnableWrapper implements Runnable {
    // 實際要執(zhí)行的線程任務(wù)
    private Runnable task;
    // 線程任務(wù)被創(chuàng)建出來的時間
    private long createTime;
    // 線程任務(wù)被線程池運(yùn)行的開始時間
    private long startTime;
    // 線程任務(wù)被線程池運(yùn)行的結(jié)束時間
    private long endTime;
    // 線程信息
    private String taskInfo;
    private boolean showWaitLog;
    /**
     * 執(zhí)行間隔時間多久,打印日志
     */
    private long durMs = 1000L;
    // 當(dāng)這個任務(wù)被創(chuàng)建出來的時候,就會設(shè)置他的創(chuàng)建時間
    // 但是接下來有可能這個任務(wù)提交到線程池后,會進(jìn)入線程池的隊列排隊
    public RunnableWrapper(Runnable task, String taskInfo) {
        this.task = task;
        this.taskInfo = taskInfo;
        this.createTime = System.currentTimeMillis();
    }
    public void setShowWaitLog(boolean showWaitLog) {
        this.showWaitLog = showWaitLog;
    }
    public void setDurMs(long durMs) {
        this.durMs = durMs;
    }
    // 當(dāng)任務(wù)在線程池排隊的時候,這個run方法是不會被運(yùn)行的
    // 但是當(dāng)任務(wù)結(jié)束了排隊,得到線程池運(yùn)行機(jī)會的時候,這個方法會被調(diào)用
    // 此時就可以設(shè)置線程任務(wù)的開始運(yùn)行時間
    @Override
    public void run() {
        this.startTime = System.currentTimeMillis();
        // 此處可以通過調(diào)用監(jiān)控系統(tǒng)的API,實現(xiàn)監(jiān)控指標(biāo)上報
        // 用線程任務(wù)的startTime-createTime,其實就是任務(wù)排隊時間
        // 這邊打印日志輸出,也可以輸出到監(jiān)控系統(tǒng)中
        if(showWaitLog) {
            log.info("任務(wù)信息: [{}], 任務(wù)排隊時間: [{}]ms", taskInfo, startTime - createTime);
        }
        // 接著可以調(diào)用包裝的實際任務(wù)的run方法
        try {
            task.run();
        } catch (Exception e) {
            log.error("run task error", e);
            throw e;
        }
        // 任務(wù)運(yùn)行完畢以后,會設(shè)置任務(wù)運(yùn)行結(jié)束的時間
        this.endTime = System.currentTimeMillis();
        // 此處可以通過調(diào)用監(jiān)控系統(tǒng)的API,實現(xiàn)監(jiān)控指標(biāo)上報
        // 用線程任務(wù)的endTime - startTime,其實就是任務(wù)運(yùn)行時間
        // 這邊打印任務(wù)執(zhí)行時間,也可以輸出到監(jiān)控系統(tǒng)中
        if(endTime - startTime > durMs) {
            log.info("任務(wù)信息: [{}], 任務(wù)執(zhí)行時間: [{}]ms", taskInfo, endTime - startTime);
        }
    }
}

使用:

我們還可以在包裝類里面封裝各種監(jiān)控行為,如本例打印日志執(zhí)行時間等。

原理探究

那大家有沒有想過為什么任務(wù)出錯會導(dǎo)致異常無法打印,甚至調(diào)度都取消了呢?讓我們從源碼出發(fā),一探究竟。

下面是調(diào)度任務(wù)的入口方法

// ScheduledThreadPoolExecutor#scheduleAtFixedRate
public ScheduledFuture<?> scheduleAtFixedRate(Runnable command,
                                              long initialDelay,
                                              long period,
                                              TimeUnit unit) {
    if (command == null || unit == null)
        throw new NullPointerException();
    if (period <= 0)
        throw new IllegalArgumentException();
    // 將執(zhí)行任務(wù)和參數(shù)包裝成ScheduledFutureTask對象
    ScheduledFutureTask<Void> sft =
        new ScheduledFutureTask<Void>(command,
                                      null,
                                      triggerTime(initialDelay, unit),
                                      unit.toNanos(period));
    RunnableScheduledFuture<Void> t = decorateTask(command, sft);
    sft.outerTask = t;
    // 延遲執(zhí)行
    delayedExecute(t);
    return t;
}
復(fù)制代碼

這個方法主要做了兩個事情

  • 將執(zhí)行任務(wù)和參數(shù)包裝成ScheduledFutureTask對象
  • 調(diào)用delayedExecute方法延遲執(zhí)行任務(wù)

延遲或周期性任務(wù)的主要執(zhí)行方法, 主要是將任務(wù)丟到隊列中,后續(xù)由工作線程獲取執(zhí)行。

// ScheduledThreadPoolExecutor#delayedExecute
private void delayedExecute(RunnableScheduledFuture<?> task) {
        if (isShutdown())
            reject(task);
        else {
            // 將任務(wù)丟到阻塞隊列中
            super.getQueue().add(task);
            if (isShutdown() &&
                !canRunInCurrentRunState(task.isPeriodic()) &&
                remove(task))
                task.cancel(false);
            else
                // 開啟工作線程,去執(zhí)行任務(wù),或者從隊列中獲取任務(wù)執(zhí)行
                ensurePrestart();
        }
    }

現(xiàn)在任務(wù)已經(jīng)在隊列中了,我們看下任務(wù)執(zhí)行的內(nèi)容是什么,還記得前面的包裝對象 ScheduledFutureTask 類,它的實現(xiàn)類是 ScheduledFutureTask ,繼承了Runnable類。

// ScheduledFutureTask#run方法
public void run() {
    // 是不是周期性任務(wù)
    boolean periodic = isPeriodic();
    if (!canRunInCurrentRunState(periodic))
        cancel(false);
    // 不是周期性任務(wù)的話, 直接調(diào)用一次下面的run    
    else if (!periodic)
        ScheduledFutureTask.super.run();
    // 如果是周期性任務(wù),則調(diào)用runAndReset方法,如果返回true,繼續(xù)執(zhí)行
    else if (ScheduledFutureTask.super.runAndReset()) {
        // 設(shè)置下次調(diào)度時間
        setNextRunTime();
        // 重新執(zhí)行調(diào)度任務(wù)
        reExecutePeriodic(outerTask);
    }
}

這里的關(guān)鍵就是看 ScheduledFutureTask.super.runAndReset() 方法是否返回true,如果是true的話繼續(xù)調(diào)度。

runAndReset方法也很簡單,關(guān)鍵就是看報異常如何處理。

// FutureTask#runAndReset
protected boolean runAndReset() {
    if (state != NEW ||
        !UNSAFE.compareAndSwapObject(this, runnerOffset,
                                     null, Thread.currentThread()))
        return false;
    // 是否繼續(xù)下次調(diào)度,默認(rèn)false
    boolean ran = false;
    int s = state;
    try {
        Callable<V> c = callable;
        if (c != null && s == NEW) {
            try {
                // 執(zhí)行任務(wù)
                c.call(); 
                // 執(zhí)行成功的話,設(shè)置為true
                ran = true;
                // 異常處理,關(guān)鍵點
            } catch (Throwable ex) {
                // 不會修改ran的值,最終是false,同時也不打印異常堆棧
                setException(ex);
            }
        }
    } finally {
        // runner must be non-null until state is settled to
        // prevent concurrent calls to run()
        runner = null;
        // state must be re-read after nulling runner to prevent
        // leaked interrupts
        s = state;
        if (s >= INTERRUPTING)
            handlePossibleCancellationInterrupt(s);
    }
    // 返回結(jié)果
    return ran && s == NEW;
}
  • 關(guān)鍵點ran變量,最終返回是不是下次繼續(xù)調(diào)度執(zhí)行
  • 如果拋出異常的話,可以看到不會修改ran為true。

總結(jié)

Java的ScheduledThreadPoolExecutor定時任務(wù)線程池所調(diào)度的任務(wù)中如果拋出了異常,并且異常沒有捕獲直接拋到框架中,會導(dǎo)致ScheduledThreadPoolExecutor定時任務(wù)不調(diào)度了。

這個結(jié)論希望大家一定要記住,不然非???,關(guān)鍵是有時候測試環(huán)境、開發(fā)環(huán)境還無法復(fù)現(xiàn),有一定的隨機(jī)性,真的到了生產(chǎn)就完蛋了。

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

關(guān)于這些知識點,我們不僅要知其然,還要知其所以然,這樣才會記憶深刻,不然很容易遺忘。

相關(guān)文章

  • Java實現(xiàn)狀態(tài)模式的示例代碼

    Java實現(xiàn)狀態(tài)模式的示例代碼

    狀態(tài)模式是一種行為型設(shè)計模式,允許對象根據(jù)其內(nèi)部狀態(tài)改變行為,本文主要介紹了Java實現(xiàn)狀態(tài)模式的示例代碼,文中通過示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2025-02-02
  • Java設(shè)計模式之中介者模式

    Java設(shè)計模式之中介者模式

    這篇文章介紹了Java設(shè)計模式之中介者模式,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-10-10
  • 創(chuàng)建Java線程安全類的七種方法

    創(chuàng)建Java線程安全類的七種方法

    線程安全是指某個方法或某段代碼,在多線程中能夠正確的執(zhí)行,不會出現(xiàn)數(shù)據(jù)不一致或數(shù)據(jù)污染的情況,我們把這樣的程序稱之為線程安全的,反之則為非線程安全的,下面這篇文章主要給大家介紹了關(guān)于創(chuàng)建Java線程安全類的七種方法,需要的朋友可以參考下
    2022-06-06
  • 使用Jenkins來構(gòu)建GIT+Maven項目的方法步驟

    使用Jenkins來構(gòu)建GIT+Maven項目的方法步驟

    這篇文章主要介紹了使用Jenkins來構(gòu)建GIT+Maven項目,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-01-01
  • 最全面的JVM優(yōu)化經(jīng)驗總結(jié)

    最全面的JVM優(yōu)化經(jīng)驗總結(jié)

    這篇文章主要介紹了最全面的JVM優(yōu)化經(jīng)驗總結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,,需要的朋友可以參考下
    2019-06-06
  • Java easyexcel導(dǎo)出報內(nèi)存溢出的問題解決

    Java easyexcel導(dǎo)出報內(nèi)存溢出的問題解決

    在Java開發(fā)時,使用EasyExcel處理大數(shù)據(jù)量導(dǎo)出可能遇到內(nèi)存溢出問題,本文深入分析了內(nèi)存溢出的原因,并提出了優(yōu)化策略,感興趣的可以了解一下
    2024-10-10
  • SpringBoot攔截器的使用

    SpringBoot攔截器的使用

    這篇文章主要給大家分享的是SpringBoot攔截器的使用,攔截器通常通過動態(tài)代理的方式來執(zhí)行。攔截器的生命周期由IoC容器管理,可以通過注入等方式來獲取其他Bean的實例,使用更方便,下面文章的詳細(xì)內(nèi)容,需要的朋友可以參考一下
    2021-11-11
  • IDEA下因Lombok插件產(chǎn)生的Library source does not match the bytecode報錯問題及解決方法(親測可用)

    IDEA下因Lombok插件產(chǎn)生的Library source does not match the bytecode報

    這篇文章主要介紹了IDEA下因Lombok插件產(chǎn)生的Library source does not match the bytecode報錯問題及解決方法,親測試過好用,需要的朋友可以參考下
    2020-04-04
  • 詳解Java爬蟲利器Jsoup

    詳解Java爬蟲利器Jsoup

    Jsoup是一款Java語言開發(fā)的HTML解析器,用于解析HTML文檔以及對HTML文檔進(jìn)行操作,處理等,本文就將詳細(xì)給大家介紹一下Java中的爬蟲利器Jsoup,感興趣的同學(xué)可以參考一下
    2023-06-06
  • 使用FeignClient調(diào)用遠(yuǎn)程服務(wù)時整合本地的實現(xiàn)方法

    使用FeignClient調(diào)用遠(yuǎn)程服務(wù)時整合本地的實現(xiàn)方法

    這篇文章主要介紹了使用FeignClient調(diào)用遠(yuǎn)程服務(wù)時整合本地的實現(xiàn)方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-03-03

最新評論