我不是Java和Springboot的专家,但我在Spring Boot中使用JDBCStore构建QuartzScheduler应用程序,我在不同的包中为不同的类使用@Configuration,@ Repository,@ Service注释,下面给出了Structure。
MySQL数据源在Controller类中可用,其中@Autowired到DataSource(它是私有的)但是就好像我在其他服务类中尝试使用相同的自动装配一样,它不起作用并且它正在抛出空指针异常。
在加载应用程序时,DataSource不可用,但稍后当我通过打印DataSource检查@PostConstruct时,它会显示HikariPool-1数据源。
能够通过RestController Get Method获取SimpleCronScheduler中的DataSource名称,就像我在TriggerListenerService中@Autowire SimpleCronScheduler一样,我无法获取对象,也无法使用其私有方法,如DataSource等。
spring.datasource.url = jdbc:mysql://localhost/quartzscheduler?useSSL=false
spring.datasource.username = root
spring.datasource.password = password
spring.datasource.driverClassName= com.mysql.jdbc.Driver
spring.quartz.jdbc.initialize-schema=never
spring.jpa.show-sql=true
spring.datasource.tomcat.testOnBorrow=true
spring.datasource.tomcat.validationQuery=SELECT 1
spring.datasource.jmx-enabled=false
spring.jpa.hibernate.ddl-auto=none
spring.jpa.properties.hibernate.dialect =
org.hibernate.dialect.MySQL5InnoDBDialect
main()方法中的DataSource为null。 DataSource的范围在config和Controller包中可用,但在services包中,它是不可访问的和Null。
我真的很困惑我在这里缺少的,请指导我,如果有任何编程概念我需要在这里学习,以获得DataSource的范围可用于整个应用程序,弹簧启动,它应该是默认的availbale但我我面临的问题是我无法使用它,因为即使在自动配置后它也会抛出空指针异常。
期待听到您的指导和建议......
------主要方法文件-----
package com.bsq.quartzscheduler;
import javax.sql.DataSource;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.scheduling.annotation.EnableScheduling;
@ComponentScan
@EnableScheduling
@SpringBootApplication
@EnableAutoConfiguration
public class QuartzSchedulerPocApplication {
@Autowired
static DataSource dataSource;
public static void main(String[] args) {
SpringApplication.run(QuartzSchedulerPocApplication.class, args);
System.out.println(" ^^^^^^^^^^ DataSource Name: "+ dataSource);
}
}
-----------------控制器文件----------
package com.bsq.quartzscheduler.controllers;
import static org.quartz.CronScheduleBuilder.cronSchedule;
import static org.quartz.JobBuilder.newJob;
import static org.quartz.TriggerBuilder.newTrigger;
import static org.quartz.impl.matchers.EverythingMatcher.allJobs;
import static org.quartz.impl.matchers.EverythingMatcher.allTriggers;
import org.quartz.CronTrigger;
import org.quartz.JobDetail;
import org.quartz.JobKey;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.quartz.impl.StdSchedulerFactory;
import org.quartz.impl.matchers.GroupMatcher;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Controller;
import com.bsq.quartzscheduler.services.JobsListenerService;
import com.bsq.quartzscheduler.services.SampleJob;
import com.bsq.quartzscheduler.services.SampleJob2;
import com.bsq.quartzscheduler.services.SampleJob3;
import com.bsq.quartzscheduler.services.TriggerListenerService;
import java.io.IOException;
import javax.annotation.PostConstruct;
import javax.sql.DataSource;
@Controller
public class SimpleCronScheduler {
private Logger logger = LoggerFactory.getLogger(getClass());
@Autowired
private JobsListenerService jobsListenerService;
@Autowired
private TriggerListenerService triggerListener;
@Autowired
private Scheduler sc;
@Autowired
private JdbcTemplate jdbcTemplate;
@Autowired
private DataSource dataSource;
@PostConstruct
public DataSource init() throws SchedulerException, IOException {
//sc = this.initiate();
logger.info("Hello from SimpleCronScheduler Controller... and DataSource is: "+ dataSource); //Outputs as HikariPool-1
return dataSource;
}
public void insertRecord() {
System.out.println(">>>>>>>DataSource : " + dataSource + "JDBC TEmplate: "+ jdbcTemplate);
String sql = "insert into qrtz_logtable (jobName, instance, createTime) values('BSQJ1', 'ANCE1','TIME')";
try {
System.out.println(">>>>>>>TRYING TO EXECUTE in insertRecord : " );
jdbcTemplate.update(sql);
} catch (Exception e) {
System.out.println(">>>>>>>exception in insertRecord : " + e.getLocalizedMessage());
}
}
@Bean
public Scheduler getScheduler() {
if ( sc == null ) {
try {
sc = StdSchedulerFactory.getDefaultScheduler();
sc.getListenerManager().addTriggerListener(triggerListener, allTriggers());
sc.getListenerManager().addJobListener(jobsListenerService, allJobs());
logger.info("Scheduler Name in getScheduler "+ sc.getSchedulerName());
} catch (SchedulerException e) {
logger.info("Exception in getScheduler method "+ e.getLocalizedMessage());
}
}
return sc;
}
@Bean
public String testMethod() {
logger.info( "Test Method Started "+dataSource);
return "TMS and DatSource Name--- "+dataSource; // Outputs as HikariPool-1 when i make a get call from RestController to this method.
}
public String startScheduler(Scheduler sc) {
try {
sc = getScheduler();
sc.start();
logger.info( "Scheduler name "+ sc.getSchedulerName()+" Started ");
return "Scheduler name "+ sc.getSchedulerName()+" Started ";
} catch (SchedulerException e) {
System.out.println("Start Scheduler exception "+ e.getMessage());
logger.info("EXCEPTION in Start Schduler "+ e.getMessage());
return "Failed to start Scheduler name "+ e.getMessage();
}
}
public String stopScheduler(){
try {
sc = getScheduler();
logger.info("Stopping Scheduler: "+ sc.getSchedulerName());
sc.standby();
logger.info("Keeping the scheduler in Standby "+ sc.getSchedulerName() );
return "Keeping the scheduler in Standby "+ sc.getSchedulerName();
} catch (SchedulerException e ) {
System.out.println("EXCEPTINO IN STOP SCHEDULER: "+ e.getMessage());
logger.info("EXCEPTION in stop Schduler "+ e.getMessage());
return "failed to Stop the the scheduler "+e.getMessage();
}
}
//Using CronTrigger to schedule the jobs
public String scheduleJobs(String jobName, int jobNum, String cronExpression, String groupName, String triggerName) {
System.out.println("Entered into Schedule Jobs");
JobDetail job = null;
CronTrigger trigger = null;
try {
sc = getScheduler();
if(jobNum==1) {
job = newJob(SampleJob.class).withIdentity(jobName, groupName).build();
} else if(jobNum ==2 ) {
job = newJob(SampleJob2.class).withIdentity(jobName, groupName).build();
} else {
job = newJob(SampleJob3.class).withIdentity(jobName, groupName).build();
}
// job = newJob(jobClass).withIdentity(jobName, groupName).build();
trigger = newTrigger().withIdentity(triggerName, groupName).withSchedule(cronSchedule(cronExpression)).build();
sc.getListenerManager().addTriggerListener(triggerListener, allTriggers());
sc.getListenerManager().addJobListener(jobsListenerService, allJobs());
sc.scheduleJob(job, trigger);
logger.info(job.getKey() + " has been scheduled to run at: " + " and repeat based on expression: "+ trigger.getCronExpression());
return "Job Added and scheduled for "+ jobName;
} catch (SchedulerException e) {
System.out.println("Failed to get Scheduler"+e.getMessage());
logger.info("Failed to get Scheduler "+ e.getMessage());
return "Failed to add JOb "+ e.getMessage();
}
}
public String runJob(String group, String jobName) {
try {
sc = getScheduler();
sc.triggerJob(new JobKey(jobName, group));
logger.info("Running the Job rightaway "+ group+"-"+jobName);
return "Triggered the Job to run now "+ group+"-"+jobName;
} catch (SchedulerException e ) {
System.out.println("EXCEPTINO IN RUN JOB : "+ e.getMessage());
logger.info("EXCEPTION in RUN JOB "+ e.getMessage());
return "Failed to run the job"+e.getMessage();
}
}
public String stopJob(String group, String jobName) {
try {
sc = getScheduler();
sc.interrupt(new JobKey(jobName, group));
logger.info("Stopping the Job rightaway "+ group+"-"+jobName);
return "Triggered the Job to run now "+ group+"-"+jobName;
} catch (SchedulerException e ) {
System.out.println("EXCEPTINO IN INTERRUPTION OF JOB : "+ e.getMessage());
logger.info("EXCEPTION INTERRUPTION OF JOB JOB "+ e.getMessage());
return "Failed to STOP the job"+e.getMessage();
}
}
public String deleteJob(String group, String name) {
try {
sc = getScheduler();
sc.deleteJob(new JobKey(name, group));
return "Job Deleted: "+ group+"-"+name;
} catch (SchedulerException e) {
return "Failed to Delete JOB: "+ group+"-"+name+ "Error is "+e.getMessage();
}
}
public String deleteAllJobs() throws SchedulerException {
sc = getScheduler();
for (String groupName : sc.getJobGroupNames()) {
for (JobKey jobKey : sc.getJobKeys(GroupMatcher.jobGroupEquals(groupName))) {
sc.deleteJob(jobKey);
}
}
return "All Jobs Deleted: ";
}
}
-------------------服务文件-----------我无法访问dataSource。
package com.bsq.quartzscheduler.services;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import javax.sql.DataSource;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.quartz.JobExecutionContext;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.quartz.Trigger;
import org.quartz.Trigger.CompletedExecutionInstruction;
import org.quartz.TriggerListener;
import org.quartz.impl.StdSchedulerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import com.bsq.quartzscheduler.controllers.SimpleCronScheduler;
@Service
public class TriggerListenerService implements TriggerListener {
private final Log logger = LogFactory.getLog(getClass());
DateFormat dateFormat = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
@Autowired
private SimpleCronScheduler scs;
@Autowired
private DataSource dataSource;
@Override
public String getName() {
return "Main Trigger(IP) Listener";
}
@Override
public void triggerComplete(Trigger trigger, JobExecutionContext context,
CompletedExecutionInstruction triggerInstructionCode) {
// do something with the event
Date date = new Date();
logger.info("Trigger Listerner###: Method: triggerComplete | Time: "+ dateFormat.format(date) +" | Job Name: " + context.getJobDetail().getKey().getName());
}
@Override
public void triggerFired(Trigger trigger, JobExecutionContext context) {
// do something with the event
Date date = new Date();
String sql = "insert into qrtz_logtable (jobName, instance, createTime) values("+context.getJobDetail().getKey().getName()+", "+context.getJobInstance()+","+dateFormat.format(date)+")";
System.out.println("DataSource in tListerner "+dataSource); //Outputs as NULL
try{
scs.insertRecord();
} catch (Exception e) {
System.out.println("ERROR MESSAGE OF SCS: "+e.getMessage()); //Outputs as NULL
}
logger.info("Trigger Listerner###: Method: triggerFired | Time: "+ dateFormat.format(date) +" | Job Name: " + context.getJobDetail().getKey().getName());
}
@Override
public void triggerMisfired(Trigger trigger) {
// do something with the event
String jName;
Date date = new Date();
try {
Scheduler sc = StdSchedulerFactory.getDefaultScheduler();
sc.getContext();
jName = sc.getJobDetail(trigger.getJobKey()).getKey().getName();
logger.info("Trigger Listerner###: Method: triggerMisfired | Time: "+ dateFormat.format(date) +" | Job Name: " + jName);
} catch (SchedulerException e) {
System.out.println("Exception in triggerMisfired " +e.getMessage());
}
System.out.println("Trigger Listener###: Method: triggerMisfired | Time:"+ dateFormat.format(date) +" | Job Name: " + trigger.getKey());
}
@Override
public boolean vetoJobExecution(Trigger trigger, JobExecutionContext context) {
// do something with the event
boolean veto = false;
Date date = new Date();
logger.info("Trigger Listerner###: Method: vetoJobExecution | Time: "+ dateFormat.format(date) +" | Job Name: " + context.getJobDetail().getKey().getName());
return veto;
}
}
我认为当您错误地声明数据源类型时会发生这种情况。
请检查Datasource类类型。
数据源应声明为标准JDBC DataSource(即javax.sql.DataSource)。
import javax.sql.DataSource;
....
@Autowired
DataSource dataSource;