1、简介
IOC的目的:控制反转和依赖注入
2、IOC过程简析(针对单例Bean)
该问题考察是否读过Spring IOC 源码,以下列举部分流程,应对面试是没问题的:
- 解析配置文件、或注解元数据,将bean的定义转换为BeanDefinition
- 调用getBean()方法获取bean。首先查找缓存,若存在,返回;否则,做创建bean的准备工作,合并BeanDefinition、实例化依赖bean
- 调用doCreateBean方法,创建bean的实例
- 通过createBeanInstance方法并根据实例化策略(Supplier、工厂方法、静态工厂方法、构造函数)例化当前bean
- 如果存在方法注入(replace_method、lookup_method),使用CGLib创建实例;否则使用反射
- 到这一步,对象已经被创建,相当于完成了控制反转
- 调用populateBean方法填充bean的属性,即依赖注入
- 解析自动装配类型,by_name、by_type
- 调用applyPropertyValue应用bean的属性
- 至此,依赖注入的工作也完成了
- 调用生命周期相关的接口,依次为(注意顺序)
- BeanNameAware 修改bean名称
- BeanFactoryAware、ApplicationContextAware 可将两者实例暴露出来,方便在其它地方获取bean的实例
- PostConstruct、InitializingBean、init_method
- DisposableBean、destroy-method
3、常见问题
3.1 三级缓存
第一级缓存:单例 singletonObjects = new ConcurrentHashMap<>(256);
第二级缓存:早期提前暴露的对象缓存 earlySingletonObjects。(属性还没有值对象也没有被初始化)
第三级缓存:singletonFactories单例对象工厂缓存
3.1 循环依赖
只能解决setter方法注入的、单例的循环依赖。通过缓存正在创建Bean实例的方式来解决循环依赖。理解循环依赖问题的关键是理解IOC的过程,反射创建对象实例后,并不能立即被使用,后续还有依赖注入、生命周期接口调用等操作。所以可以提前缓存(曝光)该bean的实例。
3.2 BeanFactory和FactoryBean
理论上讲两者之间没有什么可比性,但是比较常问。
- BeanFactory接口:IoC顶级接口,定义了bean的创建,访问等工作
- FactoryBean接口:工厂bean,该接口可以对bean进行一些额外的操作,如创建不同类型的bean、简化xml配置等。获取BeanFactory类型的bean时,如果beanName不加&则获取到对应bean的实例;如果beanName加上&,则获取到BeanFactory本身的实例