Spring загрузил компонент, но не смог получить его с помощью Autowire или ApplicationContextAware

#java #spring #spring-mvc #jakarta-ee #spring-batch

#java #spring #spring-mvc #джакарта-ee #spring-batch

Вопрос:

У меня есть компонент, который загружает XML конфигурации пакета Spring в методе PostConstruct:

     @PostConstruct
    private void starter(){            
        String[] springConfig = {
                "SpringBatch/override/application-context.xml",
                "SpringBatch/override/database.xml",
                "SpringBatch/jobs/job-dataExtraction.xml",
        };   


        ApplicationContext context = new ClassPathXmlApplicationContext(springConfig);
    }
}
  

Я использую таймер для проверки записи в базе данных и запуска задания. Таймер вызывает приведенный ниже метод в другом классе.

 public static void startUnidataJob(MessageDTO<String> msgDto){

        JobLauncher jobLauncher = (JobLauncher) ApplicationContextProvider.getApplicationContext().getBean("jobLauncher");  //Problem occurs here

            Job job = (Job) ApplicationContextProvider.getApplicationContext().getBean("dataExtractionJob");


            try {
                logger.info("With in starting...........................");
                JobExecution execution = jobLauncher.run(job, new JobParameters());

            } catch (Exception e) {
                logger.error("Job launching Exception : ", e);
            }
        }
    }
  

Проблема, когда приложение загружается, я вижу из журнала, что загружен компонент JobLauncher. Но при ApplicationContextProvider.getApplicationContext().getBean("jobLauncher") выполнении он выдает .NoSuchBeanDefinitionException: No bean named 'jobLauncher' is defined :

 2019-03-11 02:03:39,594 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Creating shared instance of singleton bean 'jobLauncher'
2019-03-11 02:03:39,594 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Creating instance of bean 'jobLauncher'
2019-03-11 02:03:39,595 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Eagerly caching bean 'jobLauncher' to allow for resolving potential circular references
2019-03-11 02:03:39,596 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Returning cached instance of singleton bean 'jobRepository'
2019-03-11 02:03:39,597 [main] DEBUG o.s.a.f.JdkDynamicAopProxy - Creating JDK dynamic proxy: target source is SingletonTargetSource for target object [org.springframework.batch.core.repository.support.SimpleJobRepository@61d84e08]
2019-03-11 02:03:39,604 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Creating instance of bean 'org.springframework.core.task.SimpleAsyncTaskExecutor#3f3ddbd9'
2019-03-11 02:03:39,605 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Finished creating instance of bean 'org.springframework.core.task.SimpleAsyncTaskExecutor#3f3ddbd9'
2019-03-11 02:03:39,605 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Invoking afterPropertiesSet() on bean with name 'jobLauncher'
2019-03-11 02:03:39,605 [main] DEBUG o.s.b.f.s.DefaultListableBeanFactory - Finished creating instance of bean 'jobLauncher'
  

Вот код ApplicationContextProvider

 @Component
public class ApplicationContextProvider implements ApplicationContextAware{

    private static ApplicationContext applicationContext;

    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
}
  

В application-context.xml который загружается через ClassPathXmlApplicationContext

 <bean id="jobLauncher"
        class="org.springframework.batch.core.launch.support.SimpleJobLauncher">
        <property name="jobRepository" ref="jobRepository" />
        <property name="taskExecutor">
            <bean class="org.springframework.core.task.SimpleAsyncTaskExecutor" />
        </property>
    </bean>

    <!-- Must set this -->
    <bean
        class="org.springframework.batch.core.configuration.support.JobRegistryBeanPostProcessor">
        <property name="jobRegistry" ref="jobRegistry" />
    </bean>

    <bean id="jobRegistry"
        class="org.springframework.batch.core.configuration.support.MapJobRegistry" />

    <bean id="taskExecutor" class="org.springframework.core.task.SimpleAsyncTaskExecutor"/>
  

Как ни странно, приложение работает нормально в моем локальном.
На сервере у меня запущено 2 одинаковых Java-приложения spring batch.
Кто-нибудь может мне помочь?

Комментарии:

1. Вы уверены, что JobLauncher был создан? Где создается «JobRepository»?

2. Описываю конкретную проблему

3. Смотрите весь мой пост. Я упоминал там @Tarun

4. Почему вы просто не внедряете зависимости в этот класс, а используете эту штуковину? Как правило, при возврате к ApplicationContextProvider основанному решению вы делаете неправильную вещь.

5. Я пробовал @Autowired, но он не может внедрить JobLauncher. Оно остается нулевым. @M

Ответ №1:

Я не вижу необходимости в ApplicationContextProvider . Вы можете внедрить job и JobLauncher в класс, в котором вы определили startUnidataJob . Например:

 public class JobScheduler {

    private static final Logger logger = LoggerFactory.getLogger(JobScheduler.class.getName());

    private Job job;

    private JobLauncher jobLauncher;

    public JobScheduler(Job job, JobLauncher jobLauncher) {
        this.job = job;
        this.jobLauncher = jobLauncher;
    }

    public void startUnidataJob(MessageDTO<String> msgDto) {
        try {
            logger.info("With in starting...........................");
            JobExecution execution = jobLauncher.run(job, new JobParameters());
        } catch (Exception e) {
            logger.error("Job launching Exception : ", e);
        }
    }
}
  

Вы можете объявить этот компонент как компонент в контексте вашего приложения, и в него будут введены job и JobLauncher. Затем вы используете его для запуска задания, когда это необходимо.

Надеюсь, это поможет.