这是我看过视频中最能解释的文字表达了
先说bean的创建过程:实例化->依赖注入->初始化
实例化之后会提前暴露到缓存,用于解决循环依赖问题。
以下的解释保证你能看懂:
为什么需要一级缓存ioc容器
总得有个地方放那些单例吧
为什么需要二级缓存
如果出现循环依赖+aop时,多个地方注入这个动态代理对象需要保证都是同一个对象,而三级缓存中的取出来的动态代理对象每次都是新对象,地址值不一样。
为什么需要三级缓存
解决循环依赖,如果出现循环依赖,先将创建好的不完整bean放入三级缓存,这样就可以给其它bean注入了。
为什么不能只使用一级缓存为什么不先判断是否是动态代理,然后直接放到一级缓存,再去注入和初始化。
一级缓存的设计就是存放完整bean对象的,如果里面有完整的,又有不完整的,违反一级缓存设计原则。
本身的设计就是先创建,再注入,再初始化然后再放入一级缓存的过程,直接放入,违反Bean的初始化的设计。
为什么不能只使用一级+三级缓存
为什么不在创建对象时,直接判断其是否需要动态代理,然后将动态代理对象直接放入三级缓存,省去中间的二级缓存。
动态代理的创建时机就是存在于初始化后,为这个bean对象生成代理对象,如果先创建的话,这样不管这个类是否存在循环依赖,都会先生成,又再一次违反Bean的初始化设计。
如果只使用这两层缓存,在使用三级缓存中的工厂对象生成的动态代理对象都是新创建的,循环依赖的时候,注入到别的bean里面去的那个动态代理对象和最终这个bean在初始化后自己创建的bean地址值不一样,或者说有2个以上的bean循环依赖的时候,他们各自拿到的bean的动态代理对象都是不一样的。所以需要一个二级缓存来存,如果二级里面有就不用查三级了。这也三级缓存和二级缓存的初始容量只有16的原因出现循环依赖本身就是代码设计不合理的,不要为了那少部分的本身不合理情况的循环依赖去改变一个大多数都合理的设计。