SpringBoot2 task scheduler 定時任務(wù)調(diào)度器四種方式
使用@EnableScheduling方式
@Component @Configurable @EnableScheduling public class Task1 { private static Log logger = LogFactory.getLog(Task1.class); @Scheduled(cron = "0/2 * * * * * ") public void execute() { logger.info("Task1>>" + new Date()); } }
xml配置方式
application 啟動加入讀取 xml 文件
@SpringBootApplication @ImportResource(value = { "classpath:applicationContext*.xml" }) public class Springboot2TaskApplication { public static void main(String[] args) { SpringApplication.run(Springboot2TaskApplication.class, args); } }
<context:component-scan base-package="com.chenyingjun.task.schedual"></context:component-scan> <task:scheduler id="appScheduler" pool-size="2" /> <!-- 調(diào)整定時任務(wù) --> <task:scheduled-tasks> <task:scheduled ref="task2" method="method2" cron="0/10 * * * * ?"/> </task:scheduled-tasks>
@Service public class Task2 { private static Log logger = LogFactory.getLog(Task2.class); public void method2() { logger.info("Task2----method2>>>>" + new Date()); } }
bean創(chuàng)建工廠方式
使用到的 jar 包
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-context-support</artifactId> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-tx</artifactId> </dependency> <dependency> <groupId>org.quartz-scheduler</groupId> <artifactId>quartz</artifactId> <version>2.3.0</version> </dependency>
新建 SchedledConfiguration.java
文件
@Configuration public class SchedledConfiguration { /** * attention: * Details:配置定時任務(wù) */ @Bean(name = "jobDetail") public MethodInvokingJobDetailFactoryBean detailFactoryBean(Task3 task) {// TestTask為需要執(zhí)行的任務(wù) MethodInvokingJobDetailFactoryBean jobDetail = new MethodInvokingJobDetailFactoryBean(); /* * 是否并發(fā)執(zhí)行 * 例如每5s執(zhí)行一次任務(wù),但是當(dāng)前任務(wù)還沒有執(zhí)行完,就已經(jīng)過了5s了, * 如果此處為true,則下一個任務(wù)會執(zhí)行,如果此處為false,則下一個任務(wù)會等待上一個任務(wù)執(zhí)行完后,再開始執(zhí)行 */ jobDetail.setConcurrent(false); // 設(shè)置任務(wù)的名字 jobDetail.setName("jobDetailName"); // 設(shè)置任務(wù)的分組,這些屬性都可以存儲在數(shù)據(jù)庫中,在多任務(wù)的時候使用 jobDetail.setGroup("jobDetailGroup"); /* * 為需要執(zhí)行的實體類對應(yīng)的對象 */ jobDetail.setTargetObject(task); /* * 通過這幾個配置,告訴JobDetailFactoryBean我們需要執(zhí)行定時執(zhí)行ScheduleTask類中的task方法 */ jobDetail.setTargetMethod("task"); return jobDetail; } /** * Details:配置定時任務(wù)的觸發(fā)器,也就是什么時候觸發(fā)執(zhí)行定時任務(wù) */ @Bean(name = "jobTrigger") public CronTriggerFactoryBean cronJobTrigger(JobDetail jobDetail) { CronTriggerFactoryBean tigger = new CronTriggerFactoryBean(); tigger.setJobDetail(jobDetail); // 初始時的cron表達式,可以改成從數(shù)據(jù)庫中獲取 tigger.setCronExpression("0/2 * * * * ?"); // trigger的name tigger.setName("tiggerName"); return tigger; } /** * Details:定義quartz調(diào)度工廠 */ @Bean(name = "scheduler") public SchedulerFactoryBean schedulerFactory(Trigger trigger) { SchedulerFactoryBean bean = new SchedulerFactoryBean(); // 用于quartz集群,QuartzScheduler 啟動時更新己存在的Job bean.setOverwriteExistingJobs(true); // 延時啟動,應(yīng)用啟動1秒后 bean.setStartupDelay(1); // 注冊觸發(fā)器 bean.setTriggers(trigger); return bean; } }
新建Task任務(wù)
@Service public class Task3 { @Resource(name = "jobTrigger") private CronTrigger cronTrigger; @Resource(name = "scheduler") private Scheduler scheduler; private static Log logger = LogFactory.getLog(Task3.class); public void task() { logger.info("Task3---------" + new Date()); } /** * 設(shè)置cron并重啟定時器 * @param cron cron值 */ public void setCron(String cron) { try { // 表達式調(diào)度構(gòu)建器 CronScheduleBuilder scheduleBuilder = CronScheduleBuilder.cronSchedule(cron); // 按新的cronExpression表達式重新構(gòu)建trigger CronTrigger trigger = (CronTrigger) scheduler.getTrigger(cronTrigger.getKey()); trigger = trigger.getTriggerBuilder().withIdentity(cronTrigger.getKey()) .withSchedule(scheduleBuilder).build(); // 按新的trigger重新設(shè)置job執(zhí)行 scheduler.rescheduleJob(cronTrigger.getKey(), trigger); } catch (SchedulerException e) { logger.info("cron表達式錯誤"); } } }
其中 setCron
方法可以進行重新設(shè)定任務(wù)調(diào)度時間
ThreadPoolTaskScheduler Runnable方式
public class Task4 implements Runnable { private ThreadPoolTaskScheduler threadPoolTaskScheduler; private static Log logger = LogFactory.getLog(Task4.class); @Override public void run() { logger.info("Task4================" + new Date()); } /** * 設(shè)置cron并啟動 * @param cronExp cron值 */ public void reStart(String cronExp) { if (null != this.threadPoolTaskScheduler) { ScheduledExecutorService scheduledExecutorService = this.threadPoolTaskScheduler.getScheduledExecutor(); if (!scheduledExecutorService.isShutdown()) { scheduledExecutorService.shutdownNow(); } this.threadPoolTaskScheduler.destroy(); } if (null != cronExp && cronExp.trim().length() > 0) { this.threadPoolTaskScheduler = new ThreadPoolTaskScheduler(); this.threadPoolTaskScheduler.setThreadNamePrefix("task4"); this.threadPoolTaskScheduler.initialize(); this.threadPoolTaskScheduler.schedule(this, new CronTrigger(cronExp)); } } }
其中reStart可以進行任務(wù)啟動和重新設(shè)置任務(wù)調(diào)度時間,調(diào)用方式如下所示
/** * 初始化task4 * 這里的代碼不應(yīng)該寫在rest層上, 應(yīng)該寫在service層上 */ @PostConstruct private void initTask4() { //初始化task4任務(wù)調(diào)度器cron,可以從數(shù)據(jù)庫中查詢到cron值 setTask4Cron("0/3 * * * * ?"); } /** * 改變task4的cron * @param cron cron值 * @return 成功標志 */ @RequestMapping(value = { "/setTask4Cron" }, method = RequestMethod.GET) @ResponseBody public String setTask4Cron(String cron) { if (null == task4) { task4 = new Task4(); } task4.reStart(cron); return "success"; }
initTask4 方法加了 @PostConstruct '注解,可以在項目啟動時自動根據(jù)cron來啟動任務(wù), setTask4Cron 方法可以重新調(diào)置任務(wù)時間
四個任務(wù)的效果如下:
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot+mail 輕松實現(xiàn)各類郵件自動推送
在實際的項目開發(fā)過程中,經(jīng)常需要用到郵件通知功能,例如,通過郵箱注冊,郵箱找回密碼,郵箱推送報表等等,實際的應(yīng)用場景非常的多,今天通過這篇文章,我們一起來學(xué)習(xí)如何在 Spring Boot 中快速實現(xiàn)一個自動發(fā)送郵件的功能2024-07-07springboot多環(huán)境進行動態(tài)配置的方法
這篇文章主要介紹了springboot多環(huán)境下如何進行動態(tài)配置,本文主要分享了如何在springboot的項目中使用多環(huán)境配置,重點是”spring.profiles.active“屬性,需要的朋友可以參考下2022-06-06SpringBoot 在IDEA中實現(xiàn)熱部署步驟詳解(實用版)
這篇文章主要介紹了SpringBoot 在IDEA中實現(xiàn)熱部署步驟詳解(實用版),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-12-12Spring注解驅(qū)動之@EventListener注解使用方式
這篇文章主要介紹了Spring注解驅(qū)動之@EventListener注解使用方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-09-09利用Java如何實現(xiàn)將二維數(shù)組轉(zhuǎn)化為鏈式儲存
鏈式結(jié)構(gòu)不要求邏輯上相鄰的節(jié)點在物理位置上也相鄰,節(jié)點間的邏輯關(guān)系是由附加的指針字段表示的,通常借助于程序設(shè)計中的指針結(jié)構(gòu)來實現(xiàn),這篇文章主要給大家介紹了關(guān)于利用Java如何實現(xiàn)將二維數(shù)組轉(zhuǎn)化為鏈式儲存的相關(guān)資料,需要的朋友可以參考下2021-12-12