您好,登錄后才能下訂單哦!
今天小編給大家分享的是springboot整合quartz項目的方法介紹,相信很多人都不太了解,為了讓大家更加了解,所以給大家總結了以下內容,一起往下看吧。一定會有所收獲的哦。
前言:quartz是一個定時調度的框架,就目前市場上來說,其實有比quartz更優秀的一些定時調度框架,不但性能比quartz好,學習成本更低,而且還提供可視化操作定時任務。例如xxl-Job,elastic-Job這兩個算是目前工作中使用比較多的定時調度框架了,適配于分布式的項目,性能也是很優秀。這是很多人就很疑惑,既然這樣我們為什么還要了解學習quartz呢?我個人覺得學習quartz有兩方面,首先xxl-Job,elastic-Job這些框架都是基于quartz的基礎上二次開發的,學習quartz更有利于我們加強理解定時調度。第二方面就是工作需求,有一些傳統互聯網公司還是有很多項目是使用quartz來完成定時任務的開發的,不懂quartz的話,老板叫你寫個定時任務都搞不定。
有上圖可以看到,一個job可以給多個jobDetail封裝,一個jobDetail可以給trigger來配置規則,但是一個trigger只能裝配一個jobDetail。
scheduler:可以理解為定時任務的工作容器或者說是工作場所,所有定時任務都是放在里面工作,可以開啟和停止。
trigger:可以理解為是定時任務任務的工作規則配置,例如說,沒個幾分鐘調用一次,或者說指定每天那個時間點執行。
jobDetail:定時任務的信息,例如配置定時任務的名字,群組之類的。
job:定時任務的真正的業務處理邏輯的地方。
這是quartz的api使用,在官網直接提供使用例子,但是在工作中用不到這種方式的
地址:https://www.quartz-scheduler.org/documentation/quartz-2.3.0/quick-start.html
public class QuartzTest { public static void main(String[] args) throws Exception{ try { Scheduler scheduler = StdSchedulerFactory.getDefaultScheduler(); scheduler.start(); JobDetail job = newJob(HelloJob.class) .withIdentity("job1", "group1") .build(); Trigger trigger = newTrigger() .withIdentity("trigger1", "group1") .startNow() .withSchedule(simpleSchedule() .withIntervalInSeconds(2) .repeatForever()) .build(); scheduler.scheduleJob(job, trigger); TimeUnit.SECONDS.sleep(20); scheduler.shutdown(); } catch (SchedulerException se) { se.printStackTrace(); } } }
在官網中介紹了,只要你引用了quartz的依賴后,springboot會自適配調度器。當然我們也可以新建bean,修改SchedulerFactoryBean的一些默認屬性值。
使用javaBean方式按實際業務需求初始化SchedulerFactoryBean(可以不要,就用默認SchedulerFactoryBean
@Configuration public class QuartzConfiguration { // Quartz配置文件路徑 private static final String QUARTZ_CONFIG = "config/quartz.properties"; @Value("${task.enabled:true}") private boolean enabled; @Autowired private DataSource dataSource; @Bean public SchedulerFactoryBean schedulerFactoryBean() { SchedulerFactoryBean schedulerFactoryBean = new SchedulerFactoryBean(); schedulerFactoryBean.setDataSource(dataSource); // 設置加載的配置文件 schedulerFactoryBean.setConfigLocation(new ClassPathResource(QUARTZ_CONFIG)); // 用于quartz集群,QuartzScheduler 啟動時更新己存在的Job schedulerFactoryBean.setOverwriteExistingJobs(true); schedulerFactoryBean.setStartupDelay(5);// 系統啟動后,延遲5s后啟動定時任務,默認為0 // 啟動時更新己存在的Job,這樣就不用每次修改targetObject后刪除qrtz_job_details表對應記錄了 schedulerFactoryBean.setOverwriteExistingJobs(true); // SchedulerFactoryBean在初始化后是否馬上啟動Scheduler,默認為true。如果設置為false,需要手工啟動Scheduler schedulerFactoryBean.setAutoStartup(enabled); return schedulerFactoryBean; } }
要使用quartz實現定時任務,首先要新建一個Job,在springboot中,新建的Job類要繼承QuartzJobBean
public class HelloJob extends QuartzJobBean { @Override protected void executeInternal(JobExecutionContext context) { StringJoiner joiner = new StringJoiner(" | ") .add("---HelloJob---") .add(context.getTrigger().getKey().getName()) .add(DateUtil.formatDate(new Date())); System.out.println(joiner); } }
創建jobDetail和Trigger來啟動定時任務,有兩種方式可以實現,本質上就是創建jobDetail和Trigger
方式一:為對應的Job創建JobDetail和Trigger,這種方式有兩個注意的地方,jobDetail一定要設置為可持久化.storeDurably(),Trigger創建要用.forJob(“helloJob”),要與JobDetail定義的相同。
@Component public class HelloJobDetailConfig { @Bean public JobDetail helloJobDetail(){ JobDetail jobDetail = JobBuilder.newJob(HelloJob.class) .withIdentity("helloJob") .storeDurably() .usingJobData("data", "保密信息") .build(); return jobDetail; } @Bean public Trigger helloJobTrigger(){ Trigger trigger = TriggerBuilder.newTrigger() .forJob("helloJob") .withSchedule(simpleSchedule() .withIntervalInSeconds(3) .repeatForever()) .build(); return trigger; } }
方式二:在注入Bean之前初始化創建JobDetail和Trigger,然后使用Scheduler來調用,跟原生API調用差不多。
@Component public class HelloJobDetailConfig2 { @Autowired private Scheduler scheduler; @PostConstruct protected void InitHelloJob() throws Exception { JobDetail jobDetail = JobBuilder.newJob(HelloJob.class) .withIdentity("helloJob") // .storeDurably() .usingJobData("data", "保密信息") .build(); Trigger trigger = TriggerBuilder.newTrigger() .withIdentity("helloTrigger") .withSchedule(simpleSchedule() .withIntervalInSeconds(3) .repeatForever()) .build(); scheduler.scheduleJob(jobDetail,trigger); } }
quartz持久化有兩種存儲,一般情況下quartz相關的表和業務表是放在同一個數據庫里的。但是如果考慮性能問題的話,就要配多數據源,業務表單獨一個庫,quartz相關的表放在一個庫。
https://docs.spring.io/spring-boot/docs/2.3.12.RELEASE/reference/html/spring-boot-features.html#boot-features-quartz
spring官網說明,默認情況下,使用內存中的JobStore。但是,如果應用程序中有DataSourcebean,并且spring.quartz是可用的,則可以配置基于JDBC的存儲。將相應地配置作業存儲類型屬性。第二個配置,每次啟動先刪除表數據再重新創建(在實際生產中,個人更傾向于拿dml來手動創建表,這個值設置為never)。在quartz的jar包里這個路徑下有不同數據庫的dml:org.quartz.impl.jdbcjobstore
spring.quartz.job-store-type=jdbc
spring.quartz.jdbc.initialize-schema=never
另外一種方式:
要讓Quartz使用DataSource而不是應用程序的主DataSource,請聲明DataSourcebean,并用@QuartzDataSource注釋其@bean方法。這樣做可以確保SchedulerFactoryBean和模式初始化都使用Quartz特定的DataSource
@Configuration public class QuartzDataSourceConfig { @Bean @QuartzDataSource public DataSource quartzDataSource() { DriverManagerDataSource dataSource = new DriverManagerDataSource(); dataSource.setUsername("root"); dataSource.setPassword("123456"); dataSource.setUrl("jdbc:mysql://127.0.0.1:3306/quartz?useUnicode=true&characterEncoding=utf-8&useSSL=false"); return dataSource; } }
還有一點需要注意:當jobbean已經注入spring容器后,下次不用需要再注入,把@Component注釋掉。
**misfire:**到了任務觸發時間點,但是任務沒有被觸發
原因:- 使用@DisallowConcurrentExecution注解,而且任務的執行時間>任務間隔
-線程池滿了,沒有資源執行任務
-機器宕機或者認為停止,果斷時間恢復運行。
@DisallowConcurrentExecution:這個是比較常用的注解,證上一個任務執行完后,再去執行下一個任務,不會允許任務并行執行。
@PersistJobDataAfterExecution:任務執行完后,會持久化保留數據到下次 執行
針對不同的ScheduleBuilder,可以設置不同的失火策略,SimpleScheduleBuilder和非SimpleScheduleBuilder,
SimpleScheduleBuilder有六種,而非SimpleScheduleBuilder有三種,在實際工作中我們使用的比較的是CronScheduleBuilder.
.withMisfireHandlingInstructionIgnoreMisfires()
MISFIRE_INSTRUCTION_IGNORE_MISFIRE_POLICY = -1
所有未觸發的執行都會立即執行,然后觸發器再按計劃運行。
.withMisfireHandlingInstructionFireAndProceed()
MISFIRE_INSTRUCTION_FIRE_ONCE_NOW = 1
立即執行第一個錯誤的執行并丟棄其他(即所有錯誤的執行合并在一起),也就是說無論錯過了多少次觸發器的執行,都只會立即執行一次,然后觸發器再按計劃運行。(默認的失火策略)
.withMisfireHandlingInstructionDoNothing()
MISFIRE_INSTRUCTION_DO_NOTHING = 2
所有未觸發的執行都將被丟棄,然后再觸發器的下一個調度周期按計劃運行。
關于quartz還有一個很重要的點就是corn表達式,這個個人認為沒必要死記硬背,實在不會寫的,上網找corn表達式在線轉換就可以了。
一個簡單demo的代碼地址:https://gitee.com/gorylee/quartz-demo
生產項目中quartz的配置使用代碼地址:https://gitee.com/gorylee/learnDemo/tree/master/quartzDemo
關于springboot整合quartz項目的方法介紹就分享到這里了,希望以上內容可以對大家有一定的參考價值,可以學以致用。如果喜歡本篇文章,不妨把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。