直接上答案
双重检查锁,两个同步锁,两次检查一级缓存
问题分析
我们先明确一些东西,什么时候是bena的不完整的,就是说我们的Bean没有完整的生命周期,只是进行了初始化或者进行了初始化和属性赋值但是没有进行实例化的不完整的bean。
首先Spring是存在三级缓存的
- 一级缓存:用来存储完整的bean
- 二级缓存:用来存储初期的不完整的bean,只进行了初始化没有进行属性赋值和实例化的bena
- 三级缓存:用来循环依赖的问题,是一个函数式接口,存放一个Bean的lambda表达式,刚实例化完的bean
Spring的解决方法
有两个线程同时访问BeanA,线程A比较快,先来访问BeanA,此时BeanA还不存在,线程A调用,getBean(A)方法,调用doGetBena(A),调用Getsingleton(A),去一级缓存二级缓存三级缓存里边都没有找到,开始创建BeanA,进行实例化,属性赋值,初始化放入一级缓存等一系列操作,这些操作Spring都是加了一把锁。 假设现在线程B进来获取BeanA,他也进行getBean(A)方法,调用doGetBena(A),调用Getsingleton(A),但是现在我们线程A在二级缓存和三级缓存加了锁,线程B现在是获取不到二级缓存和三级缓存的,处于堵塞状态, 当线程A创建完成BeanA时,删除二级缓存和三级缓存的数据,把完整的BeanA存储在一级缓存时,锁释放,然后线程B开始访问二级缓存,三级缓存,也没有访问到BeanA,线程B没有访问到BeanA,所以开始创建BeanA,在创建BeanA之前,他会再调用Getsingleton(A),去一级缓存里边拿BenaA,这时候拿到的就是完成的beanA,这就是两次检查一级缓存,从而解决了获取到不完成bean的问题。