SpringBoot使用Quartz無(wú)法注入Bean的問(wèn)題及解決
依賴
<dependency>
<groupId>org.quartz-scheduler</groupId>
<artifactId>quartz</artifactId>
<version>2.2.1</version>
</dependency>
任務(wù)工廠 JobFactory
// 解決SpringBoot不能再Q(mào)uartz中注入Bean的問(wèn)題
@Component
public class JobFactory extends AdaptableJobFactory {
/**
* AutowireCapableBeanFactory接口是BeanFactory的子類
* 可以連接和填充那些生命周期不被Spring管理的已存在的bean實(shí)例
*/
private AutowireCapableBeanFactory factory;
public JobFactory(AutowireCapableBeanFactory factory) {
this.factory = factory;
}
/**
* 創(chuàng)建Job實(shí)例
*/
@Override
protected Object createJobInstance(TriggerFiredBundle bundle) throws Exception {
// 實(shí)例化對(duì)象
Object job = super.createJobInstance(bundle);
// 進(jìn)行注入(Spring管理該Bean)
factory.autowireBean(job);
//返回對(duì)象
return job;
}
}
任務(wù)調(diào)度器 Scheduler
@Configuration
public class QuartzConfig {
private JobFactory jobFactory;
public QuartzConfig(JobFactory jobFactory){
this.jobFactory = jobFactory;
}
/**
* 配置SchedulerFactoryBean
*
* 將一個(gè)方法產(chǎn)生為Bean并交給Spring容器管理
*/
@Bean
public SchedulerFactoryBean schedulerFactoryBean() {
// Spring提供SchedulerFactoryBean為Scheduler提供配置信息,并被Spring容器管理其生命周期
SchedulerFactoryBean factory = new SchedulerFactoryBean();
// 設(shè)置自定義Job Factory,用于Spring管理Job bean
factory.setJobFactory(jobFactory);
return factory;
}
@Bean(name = "scheduler")
public Scheduler scheduler() {
return schedulerFactoryBean().getScheduler();
}
}
QuartzManager 工具類
@Service
public class QuartzManager {
private Scheduler scheduler;
public QuartzManager(Scheduler scheduler){
this.scheduler = scheduler;
}
/**
* 添加一個(gè)定時(shí)任務(wù)
*
* @param jobName 任務(wù)名
* @param jobGroupName 任務(wù)組名
* @param triggerName 觸發(fā)器名
* @param triggerGroupName 觸發(fā)器組名
* @param jobClass 任務(wù)
* @param cron 時(shí)間設(shè)置,參考quartz說(shuō)明文檔
*/
@SuppressWarnings({ "unchecked", "rawtypes" })
public void addJob(String jobName, String jobGroupName, String triggerName, String triggerGroupName, Class jobClass, String cron, Map<String, Object> params) {
try {
// 任務(wù)名,任務(wù)組,任務(wù)執(zhí)行類
JobDetail job = JobBuilder.newJob(jobClass).withIdentity(jobName, jobGroupName).build();
// 任務(wù)參數(shù)
job.getJobDataMap().putAll(params);
// 觸發(fā)器
TriggerBuilder<Trigger> triggerBuilder = TriggerBuilder.newTrigger();
// 觸發(fā)器名,觸發(fā)器組
triggerBuilder.withIdentity(triggerName, triggerGroupName);
triggerBuilder.startNow();
// 觸發(fā)器時(shí)間設(shè)定
triggerBuilder.withSchedule(CronScheduleBuilder.cronSchedule(cron));
// 創(chuàng)建Trigger對(duì)象
CronTrigger trigger = (CronTrigger) triggerBuilder.build();
// 調(diào)度容器設(shè)置JobDetail和Trigger
scheduler.scheduleJob(job, trigger);
// 啟動(dòng)
if (!scheduler.isShutdown()) {
scheduler.start();
}
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* 修改一個(gè)任務(wù)的觸發(fā)時(shí)間
*
* @param triggerName 觸發(fā)器名
* @param triggerGroupName 觸發(fā)器組名
* @param cron 時(shí)間設(shè)置,參考quartz說(shuō)明文檔
*/
public void modifyJobTime(String triggerName, String triggerGroupName, String cron) {
try {
TriggerKey triggerKey = TriggerKey.triggerKey(triggerName, triggerGroupName);
CronTrigger trigger = (CronTrigger) scheduler.getTrigger(triggerKey);
if (trigger == null) {
return;
}
String oldTime = trigger.getCronExpression();
if (!oldTime.equalsIgnoreCase(cron)) {
// 觸發(fā)器
TriggerBuilder<Trigger> triggerBuilder = TriggerBuilder.newTrigger();
// 觸發(fā)器名,觸發(fā)器組
triggerBuilder.withIdentity(triggerName, triggerGroupName);
triggerBuilder.startNow();
// 觸發(fā)器時(shí)間設(shè)定
triggerBuilder.withSchedule(CronScheduleBuilder.cronSchedule(cron));
// 創(chuàng)建Trigger對(duì)象
trigger = (CronTrigger) triggerBuilder.build();
// 方式一 :修改一個(gè)任務(wù)的觸發(fā)時(shí)間
scheduler.rescheduleJob(triggerKey, trigger);
}
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* 移除一個(gè)任務(wù)
*
* @param jobName 任務(wù)名
* @param jobGroupName 任務(wù)組名
* @param triggerName 觸發(fā)器名
* @param triggerGroupName 觸發(fā)器組名
*/
public void removeJob(String jobName, String jobGroupName, String triggerName, String triggerGroupName) {
try {
TriggerKey triggerKey = TriggerKey.triggerKey(triggerName, triggerGroupName);
// 停止觸發(fā)器
scheduler.pauseTrigger(triggerKey);
// 移除觸發(fā)器
scheduler.unscheduleJob(triggerKey);
// 刪除任務(wù)
scheduler.deleteJob(JobKey.jobKey(jobName, jobGroupName));
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* 獲取任務(wù)是否存在
*
* STATE_BLOCKED 4 阻塞
* STATE_COMPLETE 2 完成
* STATE_ERROR 3 錯(cuò)誤
* STATE_NONE -1 不存在
* STATE_NORMAL 0 正常
* STATE_PAUSED 1 暫停
*
*/
public Boolean notExists(String triggerName, String triggerGroupName) {
try {
return scheduler.getTriggerState(TriggerKey.triggerKey(triggerName, triggerGroupName)) == Trigger.TriggerState.NONE;
} catch (Exception e) {
throw new RuntimeException(e);
}
}
}
舉個(gè)栗子
自定義任務(wù) Job
@Service
public class SftpJob extends QuartzJobBean {
// 該類必須為public修飾
// 該類必須含有空參數(shù)的構(gòu)造器
@Value("${sftp.root.username}")
private String username;
@Value("${sftp.root.password}")
private String password;
@Value("${sftp.host}")
private String host;
@Value("${sftp.port}")
private Integer port;
@Autowired
private SftpRepository sftpRepository;
@Override
protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
// 傳入的參數(shù)
JobDataMap params = context.getJobDetail().getJobDataMap();
...業(yè)務(wù)邏輯...
}
}
管理任務(wù)
@Service
public class SftpTask {
// 任務(wù)名前綴
private final String job_prefix = "job_";
// 任務(wù)組前綴
private final String job_group_prefix = "job_group_";
// 觸發(fā)器前綴
private final String trigger_prefix = "trigger_";
// 觸發(fā)組前綴
private final String trigger_group_prefix = "trigger_group_";
private QuartzManager quartzManager;
public SftpTask (QuartzManager quartzManager) {
this.quartzManager = quartzManager;
}
/**
* 根據(jù)配置生成cron表達(dá)式
*/
private String getCron(SftpDTO dto) {
// 時(shí)
Integer hour = dto.getHour();
// 分
Integer minute = dto.getMinute();
// 每周幾
Integer week = dto.getWeek();
// 每月幾號(hào)
Integer day = dto.getDay();
/* 執(zhí)行時(shí)間 0每天,1每周,2每月 */
Integer execType = dto.getExecType();
String cron;
switch (execType) {
case 0:
cron = String.format("0 %s %s * * ?", minute, hour);
break;
case 1:
week = (week + 1) % 7;
cron = String.format("0 %s %s ? * %s", minute, hour, week == 0 ? 7: week);
break;
case 2:
cron = String.format("0 %s %s %s * ?", minute, hour, day);
break;
default:
cron = "0 0 0 * * ?";
break;
}
return cron;
}
/**
* 添加定時(shí)任務(wù)
*/
private void addJob(SftpDTO dto) {
Long id = dto.getId();
Map<String, Object> params = Maps.newHashMap();
params.put("id", id);
quartzManager.addJob(
job_prefix + id,
job_group_prefix + id,
trigger_prefix + id,
trigger_group_prefix + id,
SftpJob.class, getCron(etlSftpDTO), params
);
}
/**
* 修改定時(shí)任務(wù)
*/
public void modifyJob(SftpDTO dto) {
Long id = dto.getId();
if (quartzManager.notExists(trigger_prefix + id, trigger_group_prefix + id)){
// 任務(wù)不存在
addJob(dto);
} else {
// 任務(wù)存在
quartzManager.modifyJobTime(
trigger_prefix + id,
trigger_group_prefix + id,
getCron(dto)
);
}
}
/**
* 移除定時(shí)任務(wù)
*/
public void removeJob(Long id) {
quartzManager.removeJob(
job_prefix + id,
job_group_prefix + id,
trigger_prefix + id,
trigger_group_prefix + id
);
}
}
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
解決idea web工程修改js文件之后不變化的問(wèn)題
這篇文章主要介紹了解決idea web工程修改js文件之后不變化的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12
lambda表達(dá)式與傳統(tǒng)接口函數(shù)實(shí)現(xiàn)方式對(duì)比詳解
這篇文章主要為大家介紹了lambda表達(dá)式與傳統(tǒng)接口函數(shù)實(shí)現(xiàn)方式對(duì)比詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家度偶多進(jìn)步早日升職加薪2022-03-03
java面試題解LeetCode27二叉樹(shù)的鏡像實(shí)例
這篇文章主要為大家介紹了java面試題解LeetCode27二叉樹(shù)的鏡像實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
idea運(yùn)行jsp文件的時(shí)候顯示404問(wèn)題及解決
這篇文章主要介紹了idea運(yùn)行jsp文件的時(shí)候顯示404問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-10-10
Java基礎(chǔ)知識(shí)之Java語(yǔ)言概述
這篇文章主要介紹了Java基礎(chǔ)知識(shí)之Java語(yǔ)言概述,本文介紹了Java語(yǔ)言相關(guān)的基礎(chǔ)知識(shí)、歷史介紹、主要應(yīng)用方向等內(nèi)容,需要的朋友可以參考下2015-03-03
Java 實(shí)戰(zhàn)項(xiàng)目之倉(cāng)庫(kù)管理系統(tǒng)的實(shí)現(xiàn)流程
讀萬(wàn)卷書不如行萬(wàn)里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+SSM+jsp+mysql+maven實(shí)現(xiàn)一個(gè)倉(cāng)庫(kù)管理系統(tǒng),大家可以在過(guò)程中查缺補(bǔ)漏,提升水平2021-11-11
Java中的LinkedHashMap及LRU緩存機(jī)制詳解
這篇文章主要介紹了Java中的LinkedHashMap及LRU緩存機(jī)制詳解,LinkedHashMap繼承自HashMap,它的多種操作都是建立在HashMap操作的基礎(chǔ)上的,同HashMap不同的是,LinkedHashMap維護(hù)了一個(gè)Entry的雙向鏈表,保證了插入的Entry中的順序,需要的朋友可以參考下2023-09-09
Java報(bào)錯(cuò):找不到或無(wú)法加載主類的解決辦法
在Java中當(dāng)您嘗試運(yùn)行一個(gè)類作為主類時(shí),如果系統(tǒng)找不到該類或者無(wú)法加載該類,就會(huì)出現(xiàn)"找不到或無(wú)法加載主類"的錯(cuò)誤,這篇文章主要給大家介紹了關(guān)于Java報(bào)錯(cuò):找不到或無(wú)法加載主類的解決辦法,需要的朋友可以參考下2024-12-12
windows下使用 intellij idea 編譯 kafka 源碼環(huán)境
這篇文章主要介紹了使用 intellij idea 編譯 kafka 源碼的環(huán)境,本文是基于windows下做的項(xiàng)目演示,需要的朋友可以參考下2021-10-10

