Spring在进行完xml文件标签解析后,我们如何获取对应的Bean?
核心方法: getBean()
Spring三级缓存&循环依赖:
由于单例在Spring容器中只会被创建一次 ,即:创建出来的单例实例对象就会被缓存到singletonObjects(一级缓存) 中。所以,当要获得某个beanName的实例对象时,会首先尝试从singletonObjects中加载,如果加载不到,则再尝试从earlySingletonObjects(二级缓存) 中加载,如果还是加载不到,则再尝试从singletonFactories(三级缓存) 中加载对象工厂,然后通过调用其ObjectFactory#getObject() 方法来获得实例对象。(三级缓存还会通过判断Bean是否经历AOP而返回原始对象还是代理对象)
因为在创建单例bean的时候可能会存在依赖注入 的情况,所以为了避免循环依赖 ,Spring创建bean的原则是不等bean创建完成就会将创建bean的ObjectFactory提早曝光入到缓存singletonFactories中,如果下一个bean创建时需要依赖上一个bean,则直接使用ObjectFactory。(就是将还没完全初始化的非完全体的被依赖Bean给到依赖者使用)
我们以AB循环依赖为例,类A中含有属性类B,而类B中又会含有属性类A,那么初始化beanA的过程:当调用getBean(A)的时候,并不是直接去实例化A,而是先去检测缓存中是否有已经创建好的bean,或者是否已经存在创建好的ObjectFactory,而此时对于A的ObjectFactory我们早已经创建,所以便不会再去向后执行,而是直接调用ObjectFactory#getObject()方法去创建A。