AnnotationConfigApplicationContext и родительский контекст
Я столкнулся с проблемой, которая пытается определить иерархию контекста с помощью AnnotationConfigApplicationContext
.
Проблема заключается в определении контекста модуля внутри beanRefContext.xml
и установке свойства "parent" с другим контекстом (на основе XML/Annotated).
Пример:
beanRefContext.xml в модуле A
<bean id="moduleA_ApplicationContext"
class="org.springframework.context.support.ClassPathXmlApplicationContext">
<property name="configLocations">
<list>
<value>classpath:db-context.xml</value>
</list>
</property>
</bean>
DB-context.xml
<bean id="dataSource"
class="org.apache.commons.dbcp.BasicDataSource"
destroy-method="close"
p:driverClassName="org.h2.Driver"
p:url="jdbc:h2:mem:testdb;DB_CLOSE_DELAY=-1;MODE=MySQL;TRACE_LEVEL_SYSTEM_OUT=2"/>
<!-- Hibernate Session Factory -->
<bean name="sessionFactory"
class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<property name="useTransactionAwareDataSource" value="true"/>
<property name="packagesToScan">
<list>
<value>com.example.model</value>
</list>
</property>
<property name="hibernateProperties">
<!-- hibernate props -->
</property>
</bean>
beanRefContext.xml в модуле B
<bean id="moduleB_ApplicationContext"
class="org.springframework.context.annotation.AnnotationConfigApplicationContext" >
<property name="parent" ref="moduleA_ApplicationContext"/>
<constructor-arg>
<list>
<value>com.example.dao</value>
</list>
</constructor-arg>
</bean>
FooHibernateDao
class FooHibernateDao implements FooDao {
@Autowired
@Qualifier("sessionFactory")
private SessionFactory sessionsFactory;
// CRUD methods
}
В контексте приложения B не удается найти bean, определенный в контексте приложения модуля.
От взгляда на код AnnotationConfigApplicationContext
кажется, что процесс сканирования не использует родительский элемент в качестве ссылки для разрешения beans.
Есть ли что-то, что я делаю неправильно, или моя попытка создать иерархию невозможна с конфигурацией аннотаций?
Ответы
Ответ 1
Проблема связана с тем, что конструктор объекта AnnotationConfigApplicationContext выполняет проверку. Таким образом, родительский элемент не установлен на этом этапе, он устанавливается только после того, как сканирование выполняется, поскольку родительский элемент задается свойством - поэтому причина, по которой он не находит ваш bean.
По умолчанию AnnotationConfigApplicationContext bean не имеет конструктора, который принимает родительский factory - не уверен, почему.
Вы можете использовать обычный контекст приложения на основе xml и настроить сканирование аннотаций там, или вы можете создать настраиваемый fatory bean, который будет создавать контекст приложения аннотации. Это позволит указать родительскую ссылку, а затем выполнить проверку.
Взгляните на источник...
factory будет выглядеть так:
public class AnnotationContextFactory implements FactoryBean<ApplicationContext> {
private String[] packages;
private ApplicationContext parent;
@Override
public ApplicationContext getObject() throws Exception {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
context.setParent(parent);
context.scan(packages);
context.refresh();
return context;
}
@Override
public Class<ApplicationContext> getObjectType() {
return ApplicationContext.class;
}
@Override
public boolean isSingleton() {
return true;
}
public void setPackages(String... args) {
this.packages = args;
}
public void setParent(ApplicationContext parent) {
this.parent = parent;
}
}
И ваше определение bean:
<bean id="moduleB_ApplicationContext" class="za.co.test2.AnnotationContextFactory">
<property name="parent" ref="moduleA_ApplicationContext" />
<property name="packages">
<list>
<value>za.co.test2</value>
</list>
</property>
</bean>
Ответ 2
Не используйте XML для дочернего контекста.
Используйте ctx.setParent, затем ctx.register. Вот так:
public class ParentForAnnotationContextExample {
public static void main(String[] args) {
ApplicationContext parentContext = new AnnotationConfigApplicationContext(ParentContext.class);
AnnotationConfigApplicationContext childContext = new AnnotationConfigApplicationContext();
childContext.setParent(parentContext);
childContext.register(ChildContext.class); //don't add in the constructor, otherwise the @Inject won't work
childContext.refresh();
System.out.println(childContext.getBean(ParentBean.class));
System.out.println(childContext.getBean(ChildBean.class));
childContext.close();
}
@Configuration
public static class ParentContext {
@Bean ParentBean someParentBean() {
return new ParentBean();
}
}
@Configuration
public static class ChildContext {
@Bean ChildBean someChildBean() {
return new ChildBean();
}
}
public static class ParentBean {}
public static class ChildBean {
//this @Inject won't work if you use ChildContext.class in the child AnnotationConfigApplicationContext constructor
@Inject private ParentBean injectedFromParentCtx;
}
}
Ответ 3
Я столкнулся с той же проблемой,
Другой возможностью является расширение AnnotationConfigApplicationContext и добавление только необходимого конструктора или создание контекста программно, если вы создаете объект AnnotationConfigApplicationContext из java.
Ответ 4
Я сделал следующее:
BeanFactoryLocator locator = ContextSingletonBeanFactoryLocator.getInstance("classpath:beanRefContext.xml");
BeanFactoryReference parentContextRef = locator.useBeanFactory("ear.context");
ApplicationContext parentContext = (ApplicationContext) parentContextRef.getFactory();
childContext.setParent(parentContext);
И угадайте, что он работал:)
PS: Если кто-то знает, как заменить classpath: beanRefContext.xml классом @Configuration, сообщите нам всем.
Ответ 5
Я также столкнулся с подобной проблемой, и после некоторого исследования я нашел следующий подход с использованием конструктора из AnnotationConfigApplicationContext, который позволяет установить иерархию между контекстами
DefaultListableBeanFactory lbf = new DefaultListableBeanFactory(parentContext);
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(lbf);
context.register(annotatedClass1.class, annotatedClass2.class);
context.refresh();