Spring Boot中的MySQL DataSource在其他Service / Control类中不可用并抛出空指针异常

问题描述 投票:1回答:1

我不是Java和Springboot的专家,但我在Spring Boot中使用JDBCStore构建QuartzScheduler应用程序,我在不同的包中为不同的类使用@Configuration,@ Repository,@ Service注释,下面给出了Structure。

enter image description here

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;
    }

}
java spring-boot listener datasource quartz-scheduler
1个回答
0
投票

我认为当您错误地声明数据源类型时会发生这种情况。

请检查Datasource类类型。

数据源应声明为标准JDBC DataSource(即javax.sql.DataSource)。

import javax.sql.DataSource;
....
@Autowired 
DataSource dataSource;
© www.soinside.com 2019 - 2024. All rights reserved.