Как создать экземпляр spring управляемого beans во время выполнения?

Я придерживался простого рефакторинга с простого java до spring. Приложение имеет объект "Контейнер", который создает экземпляры своих частей во время выполнения. Позвольте мне объяснить с помощью кода:

public class Container {
    private List<RuntimeBean> runtimeBeans = new ArrayList<RuntimeBean>();

    public void load() {
        // repeated several times depending on external data/environment
        RuntimeBean beanRuntime = createRuntimeBean();
        runtimeBeans.add(beanRuntime);
    }

    public RuntimeBean createRuntimeBean() {
         // should create bean which internally can have some 
         // spring annotations or in other words
         // should be managed by spring
    }
}

В принципе, во время загрузки контейнер запрашивает у некоторой внешней системы информацию о количестве и конфигурации каждого RuntimeBean, а затем создает beans в соответствии с заданной спецификацией.

Проблема заключается в том, что обычно в spring

ApplicationContext context = new AnnotationConfigApplicationContext(ApplicationConfiguration.class);
Container container = (Container) context.getBean("container");

наш объект полностью настроен и содержит все зависимости. Но в моем случае я должен создать экземпляр некоторых объектов, которым также нужна инъекция зависимостей после выполнения метода load(). Как я могу это достичь?

Я использую java-config. Я уже пробовал сделать factory для RuntimeBeans:

public class BeanRuntimeFactory {

    @Bean
    public RuntimeBean createRuntimeBean() {
        return new RuntimeBean();
    }
}

ожидание @Bean для работы в так называемом режиме "lite". http://docs.spring.io/spring/docs/current/javadoc-api/org/springframework/context/annotation/Bean.html К сожалению, я не нашел разницы в простом выполнении нового RuntimeBean(); Вот сообщение с аналогичной проблемой: Как получить beans, созданный FactoryBean spring, управляемый?

Существует также http://docs.spring.io/spring/docs/current/javadoc-api/org/springframework/beans/factory/annotation/Configurable.html, но это похоже на молот в моем случае.

Я также попробовал ApplicationContext.getBean( "runtimeBean", args), где runtimeBean имеет область "Prototype", но getBean - это ужасное решение.

Upd1. Чтобы быть более конкретным, я пытаюсь реорганизовать этот класс: https://github.com/apache/lucene-solr/blob/trunk/solr/core/src/java/org/apache/solr/core/CoreContainer.java @see #load() и найти "return create (cd, false);"

UPD2. Я нашел довольно интересную вещь, названную "инъекцией метода поиска" в документации spring: http://docs.spring.io/spring/docs/current/spring-framework-reference/html/beans.html#beans-factory-lookup-method-injection

А также интересный jira билет https://jira.spring.io/browse/SPR-5192, где Фил Вебб говорит https://jira.spring.io/browse/SPR-5192?focusedCommentId=86051&page=com.atlassian.jira.plugin.system.issuetabpanels:comment-tabpanel#comment-86051, что javax.inject.Provider следует использовать здесь (это напоминает мне Guice).

Upd3. Существует также http://docs.spring.io/spring/docs/current/javadoc-api/org/springframework/beans/factory/config/ServiceLocatorFactoryBean.html

UPD4. Проблема со всеми этими методами поиска заключается в том, что они не поддерживают передачу каких-либо аргументов. Мне также нужно передать аргументы, как я делал бы с applicationContext.getBean( "runtimeBean", arg1, arg2). Похоже, в какой-то момент он был исправлен с https://jira.spring.io/browse/SPR-7431

Обновление 5. В Google Guice есть опрятная функция, называемая AssistedInject. https://github.com/google/guice/wiki/AssistedInject

Ответ 1

Похоже, я нашел решение. Поскольку я использую конфигурацию на основе java, это еще проще, чем вы можете себе представить. Альтернативным способом в xml был бы lookup-метод, однако только из spring версии 4.1.X, поскольку он поддерживает передачу аргументов методу.

Вот полный рабочий пример:

public class Container {
    private List<RuntimeBean> runtimeBeans = new ArrayList<RuntimeBean>();
    private RuntimeBeanFactory runtimeBeanFactory;

    public void load() {
        // repeated several times depending on external data/environment
        runtimeBeans.add(createRuntimeBean("Some external info1"));
        runtimeBeans.add(createRuntimeBean("Some external info2"));
    }

    public RuntimeBean createRuntimeBean(String info) {
         // should create bean which internally can have some 
         // spring annotations or in other words
         // should be managed by spring
         return runtimeBeanFactory.createRuntimeBean(info)
    }

    public void setRuntimeBeanFactory(RuntimeBeanFactory runtimeBeanFactory) {
        this.runtimeBeanFactory = runtimeBeanFactory
    }
}

public interface RuntimeBeanFactory {
    RuntimeBean createRuntimeBean(String info);
}

//and finally
@Configuration
public class ApplicationConfiguration {

    @Bean
    Container container() {
        Container container = new Container(beanToInject());
        container.setBeanRuntimeFactory(runtimeBeanFactory());
        return container;
    }

    // LOOK HOW IT IS SIMPLE IN THE JAVA CONFIGURATION
    @Bean 
    public BeanRuntimeFactory runtimeBeanFactory() {
        return new BeanRuntimeFactory() {
            public RuntimeBean createRuntimeBean(String beanName) {
                return runtimeBean(beanName);
            }
        };
    }

    @Bean
    @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    RuntimeBean runtimeBean(String beanName) {
        return new RuntimeBean(beanName);
    }
}

class RuntimeBean {
    @Autowired
    Container container;
}

Что это.

Спасибо всем.

Ответ 2

Я думаю, что ваша концепция неверна, используя   RuntimeBean beanRuntime = createRuntimeBean();
вы обходите контейнер Spring и прибегаете к использованию обычного конструктора java, поэтому любые аннотации по методу factory игнорируются, и этот bean никогда не управляется Spring

вот решение для создания нескольких прототипов beans одним способом, но не очень перспективным, но должно работать, я автоуведомленный контейнер в RuntimeBean как доказательство автопостановки, показанное в журнале, также вы можете видеть в журнале, что каждый bean является новым экземпляр прототипа при его запуске.

'

@Configuration
@ComponentScan
@EnableAutoConfiguration
public class Application {

    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);

        ApplicationContext context = new AnnotationConfigApplicationContext(Application.class);
        Container container = (Container) context.getBean("container");
        container.load();
    }
}

@Component
class Container {
    private List<RuntimeBean> runtimeBeans = new ArrayList<RuntimeBean>();
    @Autowired
    ApplicationContext context;

    @Autowired
    private ObjectFactory<RuntimeBean> myBeanFactory;

    public void load() {

        // repeated several times depending on external data/environment
        for (int i = 0; i < 10; i++) {
            // **************************************
            // COMENTED OUT THE WRONG STUFFF 
            // RuntimeBean beanRuntime = context.getBean(RuntimeBean.class);
            // createRuntimeBean();
            // 
            // **************************************

            RuntimeBean beanRuntime = myBeanFactory.getObject();
            runtimeBeans.add(beanRuntime);
            System.out.println(beanRuntime + "  " + beanRuntime.container);
        }
    }

    @Bean
    @Scope(BeanDefinition.SCOPE_PROTOTYPE)
    public RuntimeBean createRuntimeBean() {
        return new RuntimeBean();
    }
}

// @Component

class RuntimeBean {
    @Autowired
    Container container;

} '

Ответ 3

Вам не нужно Container, потому что все объекты времени выполнения должны быть созданы, удерживаться и управляться с помощью ApplicationContext. Подумайте о веб-приложении, они почти то же самое. Каждый запрос содержит информацию о внешних данных/среде, как вы упомянули выше. Что вам нужно - это прототип/запрос с областью bean, например ExternalData или EnvironmentInfo, который может считывать и удерживать данные времени выполнения через статический способ, скажем, статический метод factory.

<bean id="externalData" class="ExternalData"
    factory-method="read" scope="prototype"></bean>

<bean id="environmentInfo" class="EnvironmentInfo"
    factory-method="read" scope="prototype/singleton"></bean>

<bean class="RuntimeBean" scope="prototype">
    <property name="externalData" ref="externalData">
    <property name="environmentInfo" ref="environmentInfo">
</bean> 

Если вам нужен контейнер для сохранения объектов времени выполнения, код должен быть

class Container {

    List list;
    ApplicationContext context;//injected by spring if Container is not a prototype bean

    public void load() {// no loop inside, each time call load() will load a runtime object
        RuntimeBean bean = context.getBean(RuntimeBean.class); // see official doc
        list.add(bean);// do whatever
    }
}

Официальный doc Singleton beans с зависимостями prototype- bean.

Ответ 4

Можно динамически регистрировать bean-компоненты, используя BeanFactoryPostProcesor. Здесь вы можете сделать это, пока приложение загружается (весенний контекст приложения инициализируется). Вы не можете регистрировать beans latet, но, с другой стороны, вы можете использовать инъекцию зависимостей для ваших компонентов, поскольку они становятся "истинными" весенними бобами.

public class DynamicBeansRegistar implements BeanFactoryPostProcessor {

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        if (! (beanFactory instanceof BeanDefinitionRegistry))  {
            throw new RuntimeException("BeanFactory is not instance of BeanDefinitionRegistry);
        }   
        BeanDefinitionRegistry registry = (BeanDefinitionRegistry) beanFactory;

        // here you can fire your logic to get definition for your beans at runtime and 
        // then register all beans you need (possibly inside a loop)

        BeanDefinition dynamicBean = BeanDefinitionBuilder.    
             .rootBeanDefinition(TheClassOfYourDynamicBean.class) // here you define the class
             .setScope(BeanDefinition.SCOPE_SINGLETON)
             .addDependsOn("someOtherBean") // make sure all other needed beans are initialized

             // you can set factory method, constructor args using other methods of this builder

             .getBeanDefinition();

        registry.registerBeanDefinition("your.bean.name", dynamicBean);           

}

@Component
class SomeOtherClass {

    // NOTE: it is possible to autowire the bean
    @Autowired
    private TheClassOfYourDynamicBean myDynamicBean;

}

Как показано выше, вы все равно можете использовать Spring Dependency Injection, потому что пост-процессор работает с фактическим контекстом приложения.

Ответ 5

Простой подход:

@Component
public class RuntimeBeanBuilder {

    @Autowired
    private ApplicationContext applicationContext;

    public MyObject load(String beanName, MyObject myObject) {
        ConfigurableApplicationContext configContext = (ConfigurableApplicationContext) applicationContext;
        SingletonBeanRegistry beanRegistry = configContext.getBeanFactory();

        if (beanRegistry.containsSingleton(beanName)) {
            return beanRegistry.getSingleton(beanName);
        } else {
            beanRegistry.registerSingleton(beanName, myObject);

            return beanRegistry.getSingleton(beanName);
        }
    }
}


@Service
public MyService{

   //inject your builder and create or load beans
   @Autowired
   private RuntimeBeanBuilder builder;

   //do something
}

Вместо использования SingletonBeanRegistry вы можете использовать это:

BeanFactory beanFactory = configContext.getBeanFactory();

В любом случае SingletonBeanBuilder расширяет HierarchicalBeanFactory, а HierarchicalBeanFactory расширяет BeanFactory