一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Java教程 - Spring關(guān)于@Scheduled限制的問題

Spring關(guān)于@Scheduled限制的問題

2022-02-22 00:36辛晨V Java教程

這篇文章主要介紹了Spring關(guān)于@Scheduled限制的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

Spring @Scheduled限制

@Scheduled具有一定的限制性,它畢竟不是quartz,只是簡(jiǎn)單的定時(shí),比jdk Timer就加入了線程池而以

  • @Scheduled 不支持年份定時(shí)
  • @Scheduled 不支持W L這些字母

沒辦法 如果非要使用那就只能放棄注解使用XML方式了

Spring多定時(shí)任務(wù)@Scheduled執(zhí)行阻塞

一. 問題描述

最近項(xiàng)目中發(fā)現(xiàn)一個(gè)問題,計(jì)劃每日凌晨4:40執(zhí)行一個(gè)定時(shí)任務(wù),使用注解方式: @Scheduled(cron = “0 40 4 * * ?”),cron表達(dá)式明顯沒有問題,但是這個(gè)定時(shí)任務(wù)總是不按時(shí)執(zhí)行,有時(shí)候得等到8點(diǎn)多,有時(shí)候9點(diǎn)多才執(zhí)行。后來查了下,原來這種定時(shí)方式默認(rèn)是單線程執(zhí)行的,恰好我這里有多個(gè)定時(shí)任務(wù),并且其中有個(gè)在4:40之前的定時(shí)任務(wù)比較耗時(shí),導(dǎo)致4:40的任務(wù)只能等待之前的任務(wù)執(zhí)行完成才能夠觸發(fā),所以要自己手動(dòng)把定時(shí)任務(wù)設(shè)置成多線程的方式才行。

二. 場(chǎng)景復(fù)現(xiàn)

項(xiàng)目描述:使用Springboot進(jìn)行開發(fā)

設(shè)置兩個(gè)定時(shí)任務(wù),每5s執(zhí)行一次,并打印出其執(zhí)行情況

代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
@Component
@Log4j2
public class ScheduledTask {
    @Scheduled(cron = "0/5 * * * * ?")
    public void task1() throws InterruptedException {
        log.info("I am task11111111, current thread: {}", Thread.currentThread());
        while (true) {
            //模擬耗時(shí)任務(wù),阻塞10s
            Thread.sleep(10000);
            break;
        }
    }
    @Scheduled(cron = "0/5 * * * * ?")
    public void task2() {
        log.info("I am task22222222, current thread: {}", Thread.currentThread());
    }
}

執(zhí)行結(jié)果如下:

2019-04-24 17:11:15.008 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[scheduling-1,5,main]
2019-04-24 17:11:15.009 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[scheduling-1,5,main]
2019-04-24 17:11:25.009 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[scheduling-1,5,main]
2019-04-24 17:11:30.002 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[scheduling-1,5,main]
2019-04-24 17:11:30.003 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[scheduling-1,5,main]
2019-04-24 17:11:40.004 INFO 16868 --- [ scheduling-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[scheduling-1,5,main]

由結(jié)果可見,task1與task2由同一個(gè)線程Thread[scheduling-1,5,main]執(zhí)行,也即該定時(shí)任務(wù)默認(rèn)使用單線程,并且由于task1阻塞了10s,導(dǎo)致本應(yīng)5s執(zhí)行一次的定時(shí)任務(wù)10s才執(zhí)行一次。

三. 解決方案

網(wǎng)上有多種解決方案,以下列舉兩種

方案一:使用@Async注解實(shí)現(xiàn)異步任務(wù)

這種方式比較簡(jiǎn)單,在定時(shí)任務(wù)上加上@Async注解,注意:需啟動(dòng)類配合加上 @EnableAsync才會(huì)生效

代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
@Component
@Log4j2
public class ScheduledTask {
    @Async
    @Scheduled(cron = "0/5 * * * * ?")
    public void task1() throws InterruptedException {
        log.info("I am task11111111, current thread: {}", Thread.currentThread());
        while (true) {
            //模擬耗時(shí)任務(wù),阻塞10s
            Thread.sleep(10000);
            break;
        }
    }
    @Async
    @Scheduled(cron = "0/5 * * * * ?")
    public void task2() {
        log.info("I am task22222222, current thread: {}", Thread.currentThread());
    }
}

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

2019-04-24 17:03:00.024 INFO 2152 --- [ task-1] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[task-1,5,main]
2019-04-24 17:03:00.024 INFO 2152 --- [ task-2] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[task-2,5,main]
2019-04-24 17:03:05.001 INFO 2152 --- [ task-3] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[task-3,5,main]
2019-04-24 17:03:05.001 INFO 2152 --- [ task-4] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[task-4,5,main]
2019-04-24 17:03:10.002 INFO 2152 --- [ task-5] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[task-5,5,main]
2019-04-24 17:03:10.003 INFO 2152 --- [ task-6] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[task-6,5,main]

由運(yùn)行日志可見,定時(shí)每5s執(zhí)行一次已生效,且每次任務(wù)使用的線程不一樣,也即實(shí)現(xiàn)了多線程執(zhí)行定時(shí)任務(wù),不會(huì)出現(xiàn)任務(wù)等待現(xiàn)象。此方式據(jù)說默認(rèn)線程池大小為100,要是任務(wù)不多的話有點(diǎn)大材小用了,所以我覺得第二種方式比較好。

方案二:手動(dòng)設(shè)置定時(shí)任務(wù)的線程池大小

定時(shí)任務(wù)代碼部分還原,不使用@Async注解,新增啟動(dòng)代碼配置:

?
1
2
3
4
5
6
7
8
9
10
11
12
@Configuration
public class AppConfig implements SchedulingConfigurer {
    @Bean
    public Executor taskExecutor() {
     //指定定時(shí)任務(wù)線程數(shù)量,可根據(jù)需求自行調(diào)節(jié)
        return Executors.newScheduledThreadPool(3);
    }
    @Override
    public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
        scheduledTaskRegistrar.setScheduler(taskExecutor());
    }
}

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

2019-04-24 17:26:15.008 INFO 2164 --- [pool-1-thread-2] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[pool-1-thread-2,5,main]
2019-04-24 17:26:15.008 INFO 2164 --- [pool-1-thread-1] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[pool-1-thread-1,5,main]
2019-04-24 17:26:20.002 INFO 2164 --- [pool-1-thread-2] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[pool-1-thread-2,5,main]
2019-04-24 17:26:25.001 INFO 2164 --- [pool-1-thread-2] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[pool-1-thread-2,5,main]
2019-04-24 17:26:30.001 INFO 2164 --- [pool-1-thread-1] com.example.demo.task.ScheduledTask : I am task11111111, current thread: Thread[pool-1-thread-1,5,main]
2019-04-24 17:26:30.001 INFO 2164 --- [pool-1-thread-3] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[pool-1-thread-3,5,main]
2019-04-24 17:26:35.001 INFO 2164 --- [pool-1-thread-3] com.example.demo.task.ScheduledTask : I am task22222222, current thread: Thread[pool-1-thread-3,5,main]

由結(jié)果可見,第二種方式也實(shí)現(xiàn)了多線程任務(wù)調(diào)度。

四. 總結(jié)

兩種方式各有優(yōu)缺點(diǎn):

比較 方案一 方案二
優(yōu)點(diǎn) 注解方式使用簡(jiǎn)單,代碼量少 配置靈活,線程數(shù)可控
缺點(diǎn) 線程數(shù)不可控,可能存在資源浪費(fèi) 需要增加編碼

留個(gè)坑,從日志上看@Async方式針對(duì)同一任務(wù)也是異步的,也即task1每5s會(huì)執(zhí)行一次,但是方式二貌似對(duì)同一個(gè)任務(wù)不會(huì)生效,task1執(zhí)行的時(shí)候需等待上一次執(zhí)行結(jié)束才會(huì)觸發(fā),并沒有每5s執(zhí)行一次。關(guān)于這個(gè)現(xiàn)象,下次再琢磨…

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持服務(wù)器之家。

原文鏈接:https://blog.csdn.net/s1040342522/article/details/78275449

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: bb18lv黑料正能量 | 我不卡影院手机在线观看 | 碰91精品国产91久久婷婷 | 亚洲高清在线精品一区 | 欧美又黄又激烈真实床戏 | 国产亚洲高清国产拍精品 | 999久久久免费精品国产牛牛 | 8天堂资源在线官网 | 九九精品视频一区二区三区 | 公交车揉捏大乳呻吟喘娇 | 亚洲2卡三卡4卡5卡精品 | 亚洲男人天堂网站 | 羞羞麻豆国产精品1区2区3区 | 国产精品免费一级在线观看 | 手机看片国产自拍 | 国产精品原创巨作无遮挡 | 日本视频在线免费观看 | 男人和女人日比 | 男人猛进猛出女人下面视频 | 国产成人免费a在线资源 | 美女的隐私视频免费看软件 | 视频在线观看高清免费 | 精品国产国偷自产在线观看 | 无码人妻视频又大又粗欧美 | xx18美女美国| 国产精品国产三级国产专区不 | 国产精品久久久久久久久齐齐 | 奇米激情| 国产高清自拍视频 | 丝袜性爱 | 黑人操日本妞 | 精品视频久久久久 | ssni-497新任美脚女教师 | 美女脱了内裤让男生尿囗 | 日本68xxxxxxxxx24 日本 片 成人 在线 | 四虎新网址| 热久久天天拍天天拍热久久2018 | 99热精品在线播放 | 2020韩国三级理论在线观看 | 天天爽天天干天天操 | 九九在线免费视频 |