您好,登錄后才能下訂單哦!
springboot使用定時器@Scheduled不管用的解決方法,針對這個問題,這篇文章詳細介紹了相對應的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。
如果是一開始就不能用就是沒寫@EnableScheduling注解,如果是用著用著不管用了 是因為@Scheduled是單線程,有定時器在工作或者沒有運行完畢,所以造成了線程堵塞所以導致下一個定時器不能運行增加一個方法類
package com.llt; import org.springframework.boot.autoconfigure.batch.BatchProperties; import org.springframework.context.annotation.Configuration; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.scheduling.annotation.SchedulingConfigurer; import org.springframework.scheduling.config.ScheduledTaskRegistrar; import java.lang.reflect.Method; import java.util.concurrent.Executors; @Configuration public class ScheduleConfig implements SchedulingConfigurer { @Override public void configureTasks(ScheduledTaskRegistrar taskRegistrar) { Method[] methods = BatchProperties.Job.class.getMethods(); int defaultPoolSize = 3; int corePoolSize = 0; if (methods != null && methods.length > 0) { for (Method method : methods) { Scheduled annotation = method.getAnnotation(Scheduled.class); if (annotation != null) { corePoolSize++; } } if (defaultPoolSize > corePoolSize) corePoolSize = defaultPoolSize; } taskRegistrar.setScheduler(Executors.newScheduledThreadPool(corePoolSize)); } }
就好了!
最近項目中經常有用到@Scheduled注解,在內測時由于數據量小(沒有進行壓力測)所以每個線程執行都很快,但線上后發現部分功能無法使用,最后定位是部分的定時器沒有執行,后查閱資料和Springboot源碼后
ScheduledTaskRegistrar在啟動時,如果沒有指定線程池的大小,默認會創建核心線程數為1的默認線程池,故而當項目中出現多個@Scheduled線程時,只能一個個的執行,從而導致個別線程執行時間過長(或長期執行)時,其他定時器不能按照指定的規則進行執行。
1.在項目初始化時指定其執行線程池的大小
import org.springframework.boot.autoconfigure.batch.BatchProperties; import org.springframework.context.annotation.Configuration; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.scheduling.annotation.SchedulingConfigurer; import org.springframework.scheduling.config.ScheduledTaskRegistrar; import java.lang.reflect.Method; import java.util.Arrays; import java.util.Objects; import java.util.concurrent.ScheduledThreadPoolExecutor; import java.util.concurrent.atomic.AtomicInteger; /** * 程序名 : ScheduledTaskConfiguration * 建立日期: 2021-02-23 9:33 * 模塊 : Scheduled任務線程池設置 * 描述 : 讀取項目中使用了@Scheduled注解的方法,默認所有方法在項目創建時都需要按照設定的規則執行 * 備注 : //TODO * <p> * 修改歷史 * 序號 日期 修改人 修改原因 */ @Configuration public class ScheduledTaskConfiguration implements SchedulingConfigurer { @Override public void configureTasks(ScheduledTaskRegistrar taskRegistrar) { Method[] methods = BatchProperties.Job.class.getMethods(); final AtomicInteger corePoolSize = new AtomicInteger(); if (Objects.nonNull(methods) && methods.length > 0) { Arrays.stream(methods).forEach(method -> { final Scheduled annotation = method.getAnnotation(Scheduled.class); if (Objects.nonNull(annotation)) { corePoolSize.incrementAndGet(); } }); } ScheduledThreadPoolExecutor executor = new ScheduledThreadPoolExecutor(corePoolSize.get()); taskRegistrar.setScheduler(executor); } }
2.將定時器設置為異步線程
/** 異步線程 定時器延遲1秒啟動,每距上一次執行完成后間隔3秒執行一次 */ @Async("taskExecutor") @Scheduled(initialDelay = 1000L, fixedDelay = 3000L) public void test(){ System.out.println("---"+System.currentTimeMillis()); //業務內容 }
關于springboot使用定時器@Scheduled不管用的解決方法問題的解答就分享到這里了,希望以上內容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關注億速云行業資訊頻道了解更多相關知識。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。