缓存的实质是map
首先为什么会出现循环依赖?见下图
即在创建A类时需要填充B类的属性,创建B类时需要填充A类的属性
此时A类和B类则产生了循环依赖。
初步解决方案:
即在创建一个普通对象后,创建一个额外的Map来存储该类的普通对象,以便依赖的另一个类进行调用。但此二级缓存在初始化后的AOP就会出现问题(第五步),实质上依赖的另一个类要获取的是该类的代理对象属性,而不是普通对象属性。所以正确的二级缓存如下图所示:
即出现循环依赖后,在第一级缓存(单例池)中没有找到代理对象,则创建第二级缓存(earlySingletonObjects),并提前AOP,存放一个临时的代理对象。
二级缓存(EarlySingletonObjects)的作用:在出现循环依赖后,保证提前AOP后生成的代理对象是唯一的。其实一方面也存着没有完全属性的代理对象,因为此时的代理对象的属性还没有经过bean的完整生命周期。即二级缓存中存放的是没有经过完整的生命周期的bean
但为什么又需要三级缓存呢?
流程:第一级缓存单例池中找不到对象,则会进入到creatingSet以此出现循环依赖,然后在第二级缓存earlySingletonObjects中找,若第二级缓存找不到则会在第三级缓存singletonFactories中找,找不到则会在三级缓存中创建一个lambda表达式。这个表达式lambda若有需要处理的AOP逻辑则会提前AOP,生成一个代理对象存入二级缓存,若没有需要处理的AOP,则生成一个普通对象存入二级缓存。详情见下图所示:
各级缓存的作用:
一级缓存singletonObjects:单例池。经历一个完整的bean的生命周期之后,bean对象存入单例池。
二级缓存earlySingletonObjects:用来存储没有经历过完整的bean的生命周期的单例bean。保证单例。
三级缓存singletonFactories:存lambda表达式,某一个bean是否会出现循环依赖一开始是不知道的,三级缓存的出现让解决循环依赖的步骤更加方便。若没有三级缓存则不太好进行AOP。即存lambda表达式以进行AOP(普通对象or代理对象)
源码如下图所示: