spring浅析

目录

1、spring启动流程

2、spring bean加载过程

3、BeanFactory和FactoryBean

4、spring解决循环依赖


1、spring启动流程

  • web容器为web应用提供一个全局的上下文环境ServletContext,为IOC容器提供宿主环境。
  • web.xml中有contextLoaderListener,在web容器启动时触发容器初始化事件(ApplicationContext可以理解为IOC容器,具体实现有多个,如XmlWebApplicationContext),加载所有的service、dao、数据库配置、事务等bean。
  • 容器初始化之后,spring以WebApplicationContext.ROOTWEBAPPLICATIONCONTEXTATTRIBUTE为属性Key,将其存储到ServletContext中。
  • IOC容器初始化完毕后,开始初始化web.xml中配置的DispatcherServlet,mvc的IOC容器,加载controller的bean,并将上一步的ioc容器设置为自己的父容器。DispatcherServlet实际上是一个标准的前端控制器,用以转发、匹配、处理每个servlet请求。(个人理解:mvc的IOC容器并不需要放入ServletContext,当接收到的请求为前端url请求时走的是mvc的IOC容器,当发布webservice服务时会走ioc容器)

2、spring bean加载过程

        在IoC容器启动之后,并不会马上就实例化相应的bean,此时容器仅仅拥有所有对象的BeanDefinition(先通过扫描指定包路径下的spring注解,比如@Component、@Service、@Lazy @Sope等spring识别的注解或者是xml配置的属性(通过读取流,解析成Document)然后spring会解析这些属性,将这些属性封装到BeanDefintaion这个接口的实现类中.)。只有当getBean()调用时才是有可能触发Bean实例化阶段的活动
        实例化和初始化的区别:实例化是在jvm的堆中创建了这个对象实例,此时它只是一个空的对象,所有的属性为null。而初始化的过程就是将对象依赖的一些属性进行赋值之后,调用某些方法来开启一些默认加载。比如spring中配置的数据库属性Bean,在初始化的时候就会将这些属性填充,比如driver、jdbcurl等,然后初始化连接。

bean的生命周期可以概括为8步:

  • spring加载并实例化管理的bean
  • 将bean的引用和值注入bean的属性(依赖注入)
  • 如果bean实现了一些Aware接口注入相关内容(如果Bean实现了BeanNameAware接口,工厂调用Bean的setBeanName()方法传递Bean的ID;如果Bean实现了BeanFactoryAware接口,工厂调用setBeanFactory()方法传入工厂自身;如果Bean实现了ApplicationContextAware接口的话,Spring将调用Bean的setApplicationContext()方法,将bean所在应用上下文引用传入进来。)
  • 如果实现了BeanPostProcessor接口,调用Bean的前置处理器的postProcessBeforeInitialization()方法
  • 调用Bean的初始化方法(1.实现InitializingBean接口,继而实现afterPropertiesSet的方法 2.反射原理,配置文件使用init-method标签直接注入bean 3.@PostConstruct注解),三者的先后顺序:@PostConstruct > InitializingBean > init-method
  • 如果实现了BeanPostProcessor接口,调用Bean的后置处理器的postProcessAfterInitialization()方法
  • 使用Bean
  • 容器关闭之前,调用Bean的销毁方法

单例bean循环依赖加载过程:假设A和B循环依赖。ContextLoaderListener -> refresh() -> getBean() 开始加载bean A,首先查找一二级缓存,查到则返回bean,否则查找三级缓存,若找到执行工厂bean的getObject()方法,并将bean移动到二级缓存。若找不到则执行A的实例化(构造函数),并将不完整的bean加入三级缓存,然后开始属性注入,发现依赖B,暂停A的流程,开始先去加载B,B在缓存查不到所以也实例化并加到三级缓存,然后B开始属性注入,发现依赖A,此时A是可以在三级缓存查到的,所以移动到二级缓存,于是B可以顺利完成属性注入,并进行初始化(aware接口注入,前后置处理器,三种初始化方法),B顺利加载完成后放入一级缓存。A继续创建,完成属性注入和初始化。

3、BeanFactory和FactoryBean

  • BeanFactory是IOC容器的核心接口。Spring的容器都是它的具体实现,如:XmlBeanFactory、ApplicationContext。

  • FactoryBean是一个能生产对象的工厂bean,一个Bean A如果实现了FactoryBean接口,那么A就变成了一个工厂,根据A的名称获取到的实际上是工厂调用getObject()返回的对象,而不是A本身。

总结:他们两个都是个工厂,但FactoryBean本质上还是一个Bean,也归BeanFactory管理。BeanFactory是Spring容器的顶层接口,FactoryBean更类似于用户自定义的工厂接口。

4、spring解决循环依赖

循环依赖指两个或两个以上的bean相互依赖形成闭环,比如A依赖B,B依赖A。

  1. 构造器的循环依赖spring解决不了
  2. prototype作用域的循环依赖spring解决不了,因为spring不会缓存‘prototype’作用域的bean,而spring中循环依赖的解决正是通过缓存来实现的。
  3. setter注入构成的循环依赖:使用三级缓存。对象实例化后尽管还没有进行属性填充和初始化,但可以将不完美它提前暴露出来。在获取依赖的bean时,首先从一级缓存(存储完整的bean)中获取,获取不到再找二级缓存(存储不完整的bean),还找不到则找三级缓存(存储bean的工厂),找到后从三级缓存移除,将曝光的bean添加到二级缓存。
  • 一级缓存singletonObjects:单例对象的cache
  • 二级缓存earlySingletonObjects :提前暴光的单例对象的Cache 
  • 三级缓存singletonFactories : 单例对象工厂的cache 

总结:不要使用基于构造函数的依赖注入,可以使用

  • 基于setter方法的依赖注入
  • 在字段上使用@Autowired注解,让Spring决定在合适的时机注入(反射,暂时未深入研究)

参考文件:

Spring中BeanFactory与FactoryBean的区别

Spring 了解Bean的一生(生命周期)

Spring循环依赖的三种方式以及解决办法

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值