Spring 创建Bean的几种形式
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Spring 创建Bean的几种形式相关的知识,希望对你有一定的参考价值。
参考技术ASpring IOC 是 Spring 最重要也是最基础的两个特性之一(另外一个是 AOP ,现不在讨论)。Spring 框架的实现控制反转(IoC)的原则,也被称为依赖注入(DI)。过程对象定义它们的依赖关系,也就是说,他们使用的其它对象,只能通过构造函数参数,参数工厂方法或对象实例上设置的属性构造或从工厂回来后的方法。然后容器注入这些依赖项时创建bean。这个过程从根本上是反,因此得名“控制反转(IoC),控制实例化 bean 本身或者它的位置依赖关系通过使用直接建设类,或者一个Service Locator模式等机制。
而创建 Spring 可以通过以下几种方式:
这是从Spring最开始就支持的一种bean的配置方式。
定义实体类:
User.java ,用户类,拥有简单属性int,String类型以及复杂的对象引用Car.
Car.java
配置文件:
测试类:
定义注解Bean.
配置文件:
测试类:
通过Class来定义Bean对象。
测试类:
properties配置文件
定义实体类,并通过@ConfigurationProperties注解指定在配置文件中的前缀。
工具类:
Spring boot 启动类:并通过@EnableConfigurationProperties指定可以使用properites来实例化bean的对象类型。
测试类:
Spring-- Ioc 容器Bean实例化的几种场景
Bean实例化的几种场景
1、BeanDefinitionRegistryPostProcessor实例化:标准BeanFactoryPostProcessor的扩展,BeanFactoryPostProcessor的作用是用来进一步定义注册的BeanDefinition,IoC容器本质就是Bean管理,所以BeanFactoryPostProcessor本身也是Bean,要对BeanFactoryPostProcessor的BeanDefinition进一步定义就通过BeanDefinitionRegistryPostProcessor进行注册,BeanDefinitionRegistryPostProcessor及其子类是Ioc容器最实例化的一类Bean。它们在ConfigurableApplicationContext(ApplicationContext子接口)实现类调用refresh()方法调用invokeBeanFactoryPostProcessors(beanFactory);方法时就被实例化。
2、BeanFactoryPostProcessor实例化:仅次于BeanDefinitionRegistryPostProcessor被实例化。它们在同一个方法进行实例化,只是顺序有先后。
- PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(
- ConfigurableListableBeanFactorybeanFactory, List<BeanFactoryPostProcessor> beanFactoryPostProcessors)
3、BeanPostProcessorss实例化:BeanPostProcessors是Ioc容器Bean管理的扩展点,定义了Bean实例化前后的回调方法,那它实例化肯定是得比普通的Bean早。ConfigurableApplicationContext.registerBeanPostProcessors(beanFactory)方法负责对BeanPostProcessors进行实例化。实际上实例化BeanPostProcessor的类是PostProcessorRegistrationDelegate.registerBeanPostProcessors(beanFactory,this);
4、MessageSource类型Bean实例化:对应refresh()调用的initMessageSource();
5、ApplicationEventMulticaster(事件广播)类型Bean实例化:负责ApplicationEvent事件广播的bean,refresh()调用initApplicationEventMulticaster();进行实例化。这个Bean是不需要进行配置的,但它通用是在Ioc容器内,而且还是单例的。
6、AbstractApplicationContext类的子类可能会实例化个性化的Bean,refresh()调用onRefresh();方法做这个事情。
7、ApplicationListener类型Bean实例化:ApplicationEventMulticaster负责事件广播,它得注册广播对象,所有ApplicationListener都会在registerListeners();实例化。
1~7对应的Bean都是一些特殊的Bean,除6是由子类决定,其他的都是按照类型进行实例化,实例化调用的方法是:BeanFactory的String[] getBeanNamesForType(Class<?> type, booleanincludeNonSingletons, boolean allowEagerInit);方法,并且这些Bean设置lazy-init是无效的,设置singleton也是无效的
8、非lazy-init且singleton的Bean实例化:调用inishBeanFactoryInitialization(beanFactory);
- protected void finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory) {
- // 有容器转换服务bean时先实例化这种Bean
- if (beanFactory.containsBean(CONVERSION_SERVICE_BEAN_NAME) &&
- beanFactory.isTypeMatch(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class)) {
- beanFactory.setConversionService(
- beanFactory.getBean(CONVERSION_SERVICE_BEAN_NAME, ConversionService.class));
- }
- // 实例化LoadTimeWeaverAware类型Bean
- String[] weaverAwareNames = beanFactory.getBeanNamesForType(LoadTimeWeaverAware.class, false, false);
- for (String weaverAwareName : weaverAwareNames) {
- getBean(weaverAwareName);
- }
- // 停止使用临时类加载器
- beanFactory.setTempClassLoader(null);
- // 缓存容器中所有注册的BeanDefinition元数据,以防被修改
- beanFactory.freezeConfiguration();
- // 实例化剩余的所有非lazy-init singleton Bean
- beanFactory.preInstantiateSingletons();
- }
9、其他Bean实例化场景:a.上述1-8对应的Bean实例化依赖其他Bean时,那个Bean同时被实例化(这个在分析createBean()说明);b.延迟实例化的singleton Bean在第一次通过getBean()主动实例化它时实例化;c.prototype每次调用getBean()时都实例化;其他Scope的在它的生命周期内第一次通过getBean()主动实例化时实例化。
以上是关于Spring 创建Bean的几种形式的主要内容,如果未能解决你的问题,请参考以下文章