#java #spring #spring-boot #spring-context
Вопрос:
Поскольку мне приходится работать с устаревшим приложением, я хочу использовать компоненты из отдельного контекста Spring.
Я создаю этот отдельный контекст Spring с определением XML-компонентов, подобным этому:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd">
<context:component-scan base-package="some.different.package"/>
<bean id="connectionUrlFactory" class="my.ConnectionUrlFactory">
<property name="dbServer" value="server.url.com"/>
<property name="dbDatabasename" value="database"/>
<property name="dbClientName" value="client"/>
</bean>
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
<property name="driverClassName" value="com.sybase.jdbc4.jdbc.SybDriver"/>
<property name="url" ref="connectionUrlFactory"/>
<property name="username" value="user"/>
<property name="password" value="password"/>
</bean>
</beans>
В моем собственном «внешнем» приложении Spring Boot у меня есть такой контекст @Configuration
:
@Configuration
class Configuration {
private final ApplicationContext applicationContext;
Configuration() {
applicationContext = new ClassPathXmlApplicationContext("beans.xml");
}
@Bean
DaoSybaseImpl sybaseDao() {
return applicationContext.getBean(DaoSybaseImpl.class);
}
@Bean
DataSource sybaseDataSource() {
return applicationContext.getBean(DataSource.class);
}
}
Однако, когда я запускаю приложение, я получаю эту ошибку:
Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'sybaseDao': Unsatisfied dependency expressed through method 'setDataSource' parameter 0; nested exception is org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'javax.sql.DataSource' available: expected single matching bean but found 2: postgresDataSource,sybaseDataSource
at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredMethodElement.resolveMethodArguments(AutowiredAnnotationBeanPostProcessor.java:768)
at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredMethodElement.inject(AutowiredAnnotationBeanPostProcessor.java:720)
at org.springframework.beans.factory.annotation.InjectionMetadata.inject(InjectionMetadata.java:119)
at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessProperties(AutowiredAnnotationBeanPostProcessor.java:399)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.populateBean(AbstractAutowireCapableBeanFactory.java:1413)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:601)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:524)
at org.springframework.beans.factory.support.AbstractBeanFactory.lambda$doGetBean$0(AbstractBeanFactory.java:335)
at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:234)
at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:333)
at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:208)
at org.springframework.beans.factory.config.DependencyDescriptor.resolveCandidate(DependencyDescriptor.java:276)
at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:1380)
at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:1300)
at org.springframework.beans.factory.support.ConstructorResolver.resolveAutowiredArgument(ConstructorResolver.java:887)
at org.springframework.beans.factory.support.ConstructorResolver.createArgumentArray(ConstructorResolver.java:791)
... 125 more
Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'javax.sql.DataSource' available: expected single matching bean but found 2: postgresDataSource,sybaseDataSource
at org.springframework.beans.factory.config.DependencyDescriptor.resolveNotUnique(DependencyDescriptor.java:220)
at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:1358)
at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:1300)
at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredMethodElement.resolveMethodArguments(AutowiredAnnotationBeanPostProcessor.java:760)
... 140 more
Когда я удаляю sybaseDataSource
из своего @Configuration
, ошибка исчезает (мое приложение затем не запускается, потому что оно отсутствует sybaseDataSource
).
Похоже, что наследие ClassPathXmlApplicationContext
действительно знает о моем «внешнем» контексте загрузки Spring. Как такое может быть?
Комментарии:
1. Это просто неправильно. Бросьте свой
Configuration
класс и добавьте@IMportResource
свой@SpringBootApplication
аннотированный класс. Это сделает вашу xml-конфигурацию частью обычной конфигурации. После этого вы можете переместить компоненты из xml в java или использовать автоматическую настройку (например, для источника данных).2. @M. Deinum Я знаю, что это не идеально. Но из-за того, как построен контекст наследия, я ничего не могу сдвинуть вместе. Мне нужны две разные конфигурации унаследованного контекста.
3. Это просто не сработает, как указано, вы должны использовать
@ImportResource
Ответ №1:
Теперь я удалил «внешнее» приложение Spring. С помощью простого кода Java я могу загружать другой контекст без помех.