软件编程
位置:首页>> 软件编程>> java编程>> Spring使用 * 缓存解决循环依赖的问题

Spring使用 * 缓存解决循环依赖的问题

作者:编程の小白  发布时间:2023-03-14 09:06:15 

标签:Spring, , 缓存,循环依赖

Spring如何使用 * 缓存解决循环依赖在没开始文章之前首先来了解一下什么是循环依赖


@Component
public class A {

@Autowired
   B b;
}

@Component
public class B {

@Autowired
   A a;
}

在对象A创建过程中,需要注入B,因为容器中没有B,则去创建B,B创建过程中又需要注入A,而A在等待B的创建,B在等待A的创建,导致两者都无法创建成功,无法加入到单例池供用户使用。

Spring则通过 * 缓存来解决循环依赖的问题,另外如果对象的作用范围是Prototype,则无法通过 * 缓存解决循环依赖,会抛出BeanCurrentlyInCreationException异常,构造注入的方式,也无法解决循环依赖,只有set注入可以解决。

那么 * 缓存又是什么呢?

* 缓存就是三个Map


public class DefaultSingletonBeanRegistry extends SimpleAliasRegistry implements SingletonBeanRegistry {

//一级缓存(单例池,经过完成生命周期的对象会放入其中)
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

//二级缓存(刚实例化还未初始化的原始对象会放入其中)
   private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);

// * 缓存(存放创建某个对象的工厂)
private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

Spring Bean对象从创建到初始化大致会经过四个流程

getSingleton()doCreateBean()populateBean()addSingleton()

  • getSingleton:从单例池中获取bean对象,如果没有,则进行创建

  • doCreateBean():创建bean对象

  • populateBean():填充依赖,如果被填充的对象不存在于单例池,则进行创建等四个流程

  • addSingleton():将初始化完成的对象加入到单例池

循环依赖的对象在 * 缓存中的迁移过程

  • A 创建过程中需要 B, 于是 A 将自己放到 * 缓存里面,去实例化 B

  • B 实例化的时候发现需要 A,于是 B 先查一级缓存,没有,再查二级缓存,还是没有,再查 * 缓存

  • 找到了A,然后把 * 缓存中的 A 放到二级缓存,并删除 * 缓存中的 A

  • B 顺利初始化完毕,将自己放到一级缓存中(此时 B 中的 A 还是创建中状态,并没有完全初始化),删除 * 缓存中的 B

然后接着回来创建 A,此时 B 已经完成创建,直接从一级缓存中拿到 B,完成 A 的创建,并将 A 添加到单例池,删除二级缓存中的 A

图示:

Spring使用 * 缓存解决循环依赖的问题

来源:https://www.cnblogs.com/fkPrograming/p/14873803.html

0
投稿

猜你喜欢

手机版 软件编程 asp之家 www.aspxhome.com