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();