問題:java
問題1的簡單思考:mysql
成熟的解決方案:spring
使用方式:sql
使用@EnableScheduling註解開啓對定時任務的支持。 使用@Scheduled 註解便可,基於corn、fixedRate、fixedDelay等一些定時策略來實現定時任務。數據庫
使用缺點:springboot
使用優勢:bash
源碼分析:服務器
//默認使用的調度器
if(this.taskScheduler == null) {
this.localExecutor = Executors.newSingleThreadScheduledExecutor();
this.taskScheduler = new ConcurrentTaskScheduler(this.localExecutor);
}
//能夠看到SingleThreadScheduledExecutor指定的核心線程爲1,說白了就是單線程執行
public static ScheduledExecutorService newSingleThreadScheduledExecutor() {
return new DelegatedScheduledExecutorService
(new ScheduledThreadPoolExecutor(1));
}
//利用了DelayedWorkQueue延時隊列做爲任務的存放隊列,這樣即可以實現任務延遲執行或者定時執行
public ScheduledThreadPoolExecutor(int corePoolSize) {
super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
new DelayedWorkQueue());
}
複製代碼
使用方式:架構
方式一:由1中咱們知道之因此定時任務是阻塞執行,是配置的線程池決定的,那就好辦了,換一個不就好了!直接上代碼:併發
@Configuration
public class ScheduledConfig implements SchedulingConfigurer {
@Autowired
private TaskScheduler myThreadPoolTaskScheduler;
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
//簡單粗暴的方式直接指定
//scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
//也能夠自定義的線程池,方便線程的使用與維護,這裏很少說了
scheduledTaskRegistrar.setTaskScheduler(myThreadPoolTaskScheduler);
}
}
@Bean(name = "myThreadPoolTaskScheduler")
public TaskScheduler getMyThreadPoolTaskScheduler() {
ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler();
taskScheduler.setPoolSize(10);
taskScheduler.setThreadNamePrefix("Haina-Scheduled-");
taskScheduler.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
//調度器shutdown被調用時等待當前被調度的任務完成
taskScheduler.setWaitForTasksToCompleteOnShutdown(true);
//等待時長
taskScheduler.setAwaitTerminationSeconds(60);
return taskScheduler;
}
複製代碼
方式二:方式一的本質改變了任務調度器默認使用的線程池,接下來這種是不改變調度器的默認線程池,而是把當前任務交給一個異步線程池去執行
廢話太多,直接上代碼:
@Scheduled(fixedRate = 1000*10,initialDelay = 1000*20)
@Async("myThreadPoolTaskExecutor")
//@Async
public void scheduledTest02(){
System.out.println(Thread.currentThread().getName()+"--->xxxxx--->"+Thread.currentThread().getId());
}
//自定義線程池
@Bean(name = "myThreadPoolTaskExecutor")
public TaskExecutor getMyThreadPoolTaskExecutor() {
ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
taskExecutor.setCorePoolSize(20);
taskExecutor.setMaxPoolSize(200);
taskExecutor.setQueueCapacity(25);
taskExecutor.setKeepAliveSeconds(200);
taskExecutor.setThreadNamePrefix("Haina-ThreadPool-");
// 線程池對拒絕任務(無線程可用)的處理策略,目前只支持AbortPolicy、CallerRunsPolicy;默認爲後者
taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
//調度器shutdown被調用時等待當前被調度的任務完成
taskExecutor.setWaitForTasksToCompleteOnShutdown(true);
//等待時長
taskExecutor.setAwaitTerminationSeconds(60);
taskExecutor.initialize();
return taskExecutor;
}
複製代碼
線程池的使用心得
問題:
使用@Scheduled註解來完成設置定時任務,可是有時候咱們每每須要對週期性的時間的設置會作一些改變,或者要動態的啓停一個定時任務,那麼這個時候使用此註解就不太方便了,緣由在於這個註解中配置的cron表達式必須是常量,那麼當咱們修改定時參數的時候,就須要中止服務,從新部署。
解決辦法: 方式一:實現SchedulingConfigurer接口,重寫configureTasks方法,從新制定Trigger,核心方法就是addTriggerTask(Runnable task, Trigger trigger) ,不過須要注意的是,此種方式修改了配置值後,須要在下一次調度結束後,纔會更新調度器,並不會在修改配置值時實時更新,實時更新須要在修改配置值時額外增長相關邏輯處理。
@Configuration
public class ScheduledConfig implements SchedulingConfigurer {
@Autowired
private TaskScheduler myThreadPoolTaskScheduler;
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
//scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
scheduledTaskRegistrar.setTaskScheduler(myThreadPoolTaskScheduler);
//能夠實現動態調整定時任務的執行頻率
scheduledTaskRegistrar.addTriggerTask(
//1.添加任務內容(Runnable)
() -> System.out.println("cccccccccccccccc--->" + Thread.currentThread().getId()),
//2.設置執行週期(Trigger)
triggerContext -> {
//2.1 從數據庫動態獲取執行週期
String cron = "0/2 * * * * ? ";
//2.2 合法性校驗.
// if (StringUtils.isEmpty(cron)) {
// // Omitted Code ..
// }
//2.3 返回執行週期(Date)
return new CronTrigger(cron).nextExecutionTime(triggerContext);
}
);
}
}
複製代碼
方式二:使用threadPoolTaskScheduler類可實現動態添加刪除功能,固然也可實現執行頻率的調整
首先,咱們要認識下這個調度類,它實際上是對java中ScheduledThreadPoolExecutor的一個封裝改進後的產物,主要改進有如下幾點:
順便說下ThreadPoolTaskExecutor相對於ThreadPoolExecutor的改進點:
扯了這麼多,仍是直接上代碼:
@Component
public class DynamicTimedTask {
private static final Logger logger = LoggerFactory.getLogger(DynamicTimedTask.class);
//利用建立好的調度類統一管理
//@Autowired
//@Qualifier("myThreadPoolTaskScheduler")
//private ThreadPoolTaskScheduler myThreadPoolTaskScheduler;
//接受任務的返回結果
private ScheduledFuture<?> future;
@Autowired
private ThreadPoolTaskScheduler threadPoolTaskScheduler;
//實例化一個線程池任務調度類,可使用自定義的ThreadPoolTaskScheduler
@Bean
public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
ThreadPoolTaskScheduler executor = new ThreadPoolTaskScheduler();
return new ThreadPoolTaskScheduler();
}
/**
* 啓動定時任務
* @return
*/
public boolean startCron() {
boolean flag = false;
//從數據庫動態獲取執行週期
String cron = "0/2 * * * * ? ";
future = threadPoolTaskScheduler.schedule(new CheckModelFile(),cron);
if (future!=null){
flag = true;
logger.info - 最佳的logger 來源和相關信息。("定時check訓練模型文件,任務啓動成功!!!");
}else {
logger.info - 最佳的logger 來源和相關信息。("定時check訓練模型文件,任務啓動失敗!!!");
}
return flag;
}
/**
* 中止定時任務
* @return
*/
public boolean stopCron() {
boolean flag = false;
if (future != null) {
boolean cancel = future.cancel(true);
if (cancel){
flag = true;
logger.info - 最佳的logger 來源和相關信息。("定時check訓練模型文件,任務中止成功!!!");
}else {
logger.info - 最佳的logger 來源和相關信息。("定時check訓練模型文件,任務中止失敗!!!");
}
}else {
flag = true;
logger.info - 最佳的logger 來源和相關信息。("定時check訓練模型文件,任務已經中止!!!");
}
return flag;
}
class CheckModelFile implements Runnable{
@Override
public void run() {
//編寫你本身的業務邏輯
System.out.print("模型文件檢查完畢!!!")
}
}
}
複製代碼
到此基於springtask下的定時任務的簡單使用算是差很少了,其中難免有些錯誤的地方,或者理解有偏頗的地方歡迎你們提出來!
分享免費學習資料
針對於還會準備免費的Java架構學習資料(裏面有高可用、高併發、高性能及分佈式、Jvm性能調優、MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多個知識點的架構資料) 爲何某些人會一直比你優秀,是由於他自己就很優秀還一直在持續努力變得更優秀,而你是否是還在知足於現狀心裏在竊喜!但願讀到這的您能點個小贊和關注下我,之後還會更新技術乾貨,謝謝您的支持!
資料領取方式:加入粉絲羣963944895
,私信管理員便可免費領取