Spring boot - один @Configuration @import будет работать, а несколько - нет, почему? - PullRequest
1 голос
/ 09 марта 2020

В загрузочном проекте Spring у меня есть RestController, который содержит:

@Autowired
private ProactiveDbManager proactiveDbManager;

@Autowired
private CoreDbManager coreDbManager;

Тезисы менеджеров 2 дБ находятся в другом загрузочном модуле Spring, поэтому я импортирую их с классом конфигурации:

@Configuration
@PropertySource("classpath:application.properties")
@Import({ProactiveDbCommonsConfig.class})
public class ProactiveDbConfig {

}

и:

@Configuration
@PropertySource("classpath:application.properties")
@Import({CoreDbCommonsConfig.class})
public class CoreDbConfig {

}

ProactiveDbCommonsConfig и CoreDbCommonsConfig также являются конфигурациями. ProactiveDbCommonsConfig:

@Configuration
public class ProactiveDbCommonsConfig {
    private static final Logger LOGGER = LoggerFactory.getLogger(ProactiveDbCommonsConfig.class);

    private ProactiveDbManager proactiveDbManager;
    @Bean
    public MasterHandler masterHandler() {
        return new MasterHandler(true);
    }

    @Bean
    public ProactiveDbManager dBManager(MasterHandler masterHandler) {
        List<String> dbConnectionStrings = masterHandler.getDbConnectionStringsJsonsList(DbConsts.DEFAULT_DATA_CENTER);
        LOGGER.debug("DB manager created.");
        proactiveDbManager = new ProactiveDbManager(dbConnectionStrings);
        return proactiveDbManager;
    }


    @Override
    protected void finalize() {
        try {
            closeDbConnection();
        } catch (Exception ignored) {
        }
    }

    @PreDestroy
    public void closeDbConnection() throws Exception {
        if (this.proactiveDbManager != null) {
            this.proactiveDbManager.disconnect();
            this.proactiveDbManager = null;
        }
    }
}

Моя проблема заключается в том, что, если я удалю 1 из автопроводки в контроллере и удалю его конфигурацию (то есть ProactiveDbConfig), то это сработает. Они оба работают в одиночку, но не вместе.

Я получаю ошибку:

Field coreDbManager in x.y.z.controllers.h required a bean of type 'x.y.a.db.managers.CoreDbManager' that could not be found.

Трассировка стека:

15:56:14.609 [main] DEBUG o.s.b.d.LoggingFailureAnalysisReporter - Application failed to start due to an exception
org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'x.y.db.managers.CoreDbManager' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.raiseNoMatchingBeanFound(DefaultListableBeanFactory.java:1493)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:1104)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:1066)
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:585)
    at org.springframework.beans.factory.annotation.InjectionMetadata.inject(InjectionMetadata.java:88)
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessPropertyValues(AutowiredAnnotationBeanPostProcessor.java:366)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.populateBean(AbstractAutowireCapableBeanFactory.java:1264)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:553)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:483)
    at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:306)
    at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:230)
    at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:302)
    at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:197)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.preInstantiateSingletons(DefaultListableBeanFactory.java:761)
    at org.springframework.context.support.AbstractApplicationContext.finishBeanFactoryInitialization(AbstractApplicationContext.java:867)
    at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:543)
    at org.springframework.boot.context.embedded.EmbeddedWebApplicationContext.refresh(EmbeddedWebApplicationContext.java:122)
    at org.springframework.boot.SpringApplication.refresh(SpringApplication.java:693)
    at org.springframework.boot.SpringApplication.refreshContext(SpringApplication.java:360)
    at org.springframework.boot.SpringApplication.run(SpringApplication.java:303)
    at org.springframework.boot.SpringApplication.run(SpringApplication.java:1118)
    at org.springframework.boot.SpringApplication.run(SpringApplication.java:1107)
    at x.y.app.MyApp.main(MyApp.java:10)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at com.intellij.rt.execution.CommandLineWrapper.main(CommandLineWrapper.java:66)
15:56:14.609 [main] ERROR o.s.b.d.LoggingFailureAnalysisReporter - 

1 Ответ

2 голосов
/ 09 марта 2020

При использовании @Configuration классов и без явного определения имени компонента в @Bean(name = "<your-name-here>") имя метода будет использоваться в качестве имени компонента.

Если вы импортируете несколько файлов конфигурации, которые содержат компоненты с одинаковым именем, один переопределит другой. Новые версии Spring Boot предотвращают это, отключая переопределение (которое включено по умолчанию), что приведет к исключению при запуске.

В вашем случае решение, либо переименуйте метод, либо добавьте явное имя к аннотации @Bean.

...