spring 如何解决循环依赖

首先解释下什么是循环依赖,其实很简单,就是有两个类它们互相都依赖了对方,如下所示:

@Component
public class AService {

  @Autowired
  private BService bService;
}
@Component
public class BService {

  @Autowired
  private AService aService;
}

AService和BService显然两者都在内部依赖了对方,单拎出来看仿佛看到了多线程中常见的死锁代码,但很显然Spring解决了这个问题,不然我们也不可能正常的使用它了。

所谓创建Bean实际上就是调用getBean() 方法,这个方法可以在AbstractBeanFactory这个类里面找到,这个方法一开始会调用getSingleton()方法。

// Eagerly check singleton cache for manually registered singletons.
Object sharedInstance = getSingleton(beanName);

这个方法的实现长得很有意思,有着一堆if语句。

protected Object getSingleton(String beanName, boolean allowEarlyReference) {
  Object singletonObject = this.singletonObjects.get(beanName);
  if (singletonObject == null && this.isSingletonCurrentlyInCreation(beanName)) {
    singletonObject = this.earlySingletonObjects.get(beanName);
    if (singletonObject == null && allowEarlyReference) {
      synchronized(this.singletonObjects) {
        singletonObject = this.singletonObjects.get(beanName);
        if (singletonObject == null) {
          singletonObject = this.earlySingletonObjects.get(beanName);
          if (singletonObject == null) {
            ObjectFactory<?> singletonFactory = (ObjectFactory)this.singletonFactories.get(beanName);
            if (singletonFactory != null) {
              singletonObject = singletonFactory.getObject();
              // 从三级缓存里取出放到二级缓存中
              this.earlySingletonObjects.put(beanName, singletonObject);
              this.singletonFactories.remove(beanName);
            }
          }
        }
      }
    }
  }

  return singletonObject;
}

但这一坨if很好理解,就是一层层的去获取这个bean,首先从singletonObjects中获取,这里面存放的是已经完全创建好的单例Bean;如果取不到,那么就往下走,去earlySingletonObjects里面取,这个是早期曝光的对象;如果还是没有,那么再去第三级缓存singletonFactories里面获取,它是提前暴露的对象工厂,这里会从三级缓存里取出后放到二级缓存中。那么总的来说,Spring去获取一个bean的时候,其实并不是直接就从容器里面取,而是先从缓存里找,而且缓存一共有三级。那么从这个方法返回的并不一定是我们需要的bean,后面会调用getObjectForBeanInstance()方法去得到实例化后的bean,这里就不多说了。

但如果缓存里面的确是取不到bean呢?那么说明这个bean的确还未创建,需要去创建一个bean,这样我们就会去到前一篇生命周期中的创建bean的方法了。回顾下流程:实例化–属性注入–初始化–销毁。那么我们回到文章开头的例子,有ServiceA和ServiceB两个类。一般来说,Spring是按照自然顺序去创建bean,那么第一个要创建的是ServiceA。显然一开始缓存里是没有的,我们会来到创建bean的方法。首先进行实例化阶段,我们会来到第一个跟解决循环依赖有关的代码,在实例化阶段的代码中就可以找到。

// Eagerly cache singletons to be able to resolve circular references
// even when triggered by lifecycle interfaces like BeanFactoryAware.
boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
   isSingletonCurrentlyInCreation(beanName));
if (earlySingletonExposure) {
  if (logger.isTraceEnabled()) {
   logger.trace("Eagerly caching bean '" + beanName +
      "' to allow for resolving potential circular references");
  }
  addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}

首先看看第一行,earlySingletonExposure这个变量它会是什么值?

它是有一个条件表达式返回的,一个个来看,首先,mbd.isSingleton()。我们知道Spring默认的Bean的作用域都是单例的,因此这里正常来说都是返回true没问题。第二个,this.allowCircularReference,这个变量是标记是否允许循环引用,默认也是true。第三个,调用了一个方法,isSingletonCurrentlyInCreation(beanName),进入该代码可以看出它是返回当前的bean是不是正常创建,显然也是true。因此这个earlySingletonExposure返回的就是true。

接下来就进入了if语句的实现里面了,也就是addSingletonFactory()这个方法。看到里面的代码中出现singletonFactories这个变量是不是很熟悉?翻到上面的getSingleton()就知道了,其实就是三级缓存,所以这个方法的作用是通过三级缓存提前暴露一个工厂对象。

/**
 * Add the given singleton factory for building the specified singleton
 * if necessary.
 * <p>To be called for eager registration of singletons, e.g. to be able to
 * resolve circular references.
 * @param beanName the name of the bean
 * @param singletonFactory the factory for the singleton object
 */
protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
  Assert.notNull(singletonFactory, "Singleton factory must not be null");
  synchronized (this.singletonObjects) {
   if (!this.singletonObjects.containsKey(beanName)) {
     this.singletonFactories.put(beanName, singletonFactory);
     this.earlySingletonObjects.remove(beanName);
     this.registeredSingletons.add(beanName);
   }
  }
}

接下来,回忆下上一章节说的实例化之后的步骤,就是属性注入了。这就意味着ServiceA需要将ServiceB注入进去,那么显然又要调用getBean()方法去获取ServiceB。ServiceB还没有创建,则也会进入这个createBean()方法,同样也会来到这一步依赖注入。ServiceB中依赖了ServiceA,则会调用getBean()去获取ServiceA。此时的获取ServiceA可就不是再创建Bean了,而是从缓存中获取。这个缓存就是上面getSingleton()这个方法里面我们看到的singletonFactory。那么这个singletonFactory哪里来的,就是这个addSingletonFactory()方法的第二个参数,即getEarlyBeanReference()方法。

/**
 * Obtain a reference for early access to the specified bean,
 * typically for the purpose of resolving a circular reference.
 * @param beanName the name of the bean (for error handling purposes)
 * @param mbd the merged bean definition for the bean
 * @param bean the raw bean instance
 * @return the object to expose as bean reference
 */
protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) {
  Object exposedObject = bean;
  if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
   for (SmartInstantiationAwareBeanPostProcessor bp : getBeanPostProcessorCache().smartInstantiationAware) {
     exposedObject = bp.getEarlyBeanReference(exposedObject, beanName);
   }
  }
  return exposedObject;
}

查看bp.getEarlyBeanReference(exposedObject, beanName)的实现,发现有两个,一个是spring-beans下的SmartInstantiationAwareBeanPostProcessor,一个是spring-aop下的AbstractAutoProxyCreator。我们在未使用AOP的情况下,取的还是第一种实现。

default Object getEarlyBeanReference(Object bean, String beanName) throws BeansException {
  return bean;
}

那么令人惊讶的是,这方法直接返回了bean,也就是说如果不考虑AOP的话,这个方法啥都没干,就是把实例化创建的对象直接返回了。如果考虑AOP的话调用的是另一个实现:

public Object getEarlyBeanReference(Object bean, String beanName) {
  Object cacheKey = getCacheKey(bean.getClass(), beanName);
  this.earlyProxyReferences.put(cacheKey, bean);
  return wrapIfNecessary(bean, beanName, cacheKey);
}

可以看出,如果使用了AOP的话,这个方法返回的实际上是bean的代理,并不是它本身。那么通过这部分我们可以认为,在没有使用AOP的情况下,三级缓存是没有什么用的,所谓三级缓存实际上只是跟Spring的AOP有关的。

好了我们现在是处于创建B的过程,但由于B依赖A,所以调用了获取A的方法,则A从三级缓存进入了二级缓存,得到了A的代理对象。当然我们不需要担心注入B的是A的代理对象会带来什么问题,因为生成代理类的内部都是持有一个目标类的引用,当调用代理对象的方法的时候,实际上是会调用目标对象的方法的,所以所以代理对象是没影响的。当然这里也反应了我们实际上从容器中要获取的对象实际上是代理对象而不是其本身。

那么我们再回到创建A的逻辑往下走,能看到后面实际上又调用了一次getSingleton()方法。传入的allowEarlyReference为false。

if (earlySingletonExposure) {
  Object earlySingletonReference = getSingleton(beanName, false);
  if (earlySingletonReference != null) {
   if (exposedObject == bean) {
     exposedObject = earlySingletonReference;
   }
   ...
  }
}

翻看上面的getSingleton()代码可以看出,allowEarlyReference为false就相当于禁用三级缓存,代码只会执行到通过二级缓存get。

singletonObject = this.earlySingletonObjects.get(beanName);

因为在前面我们在创建往B中注入A的时候已经从三级缓存取出来放到二级缓存中了,所以这里A可以通过二级缓存去取。再往下就是生命周期后面的代码了,就不再继续了。

那么现在就会有个疑问,我们为什么非要三级缓存,直接用二级缓存似乎就足够了?

看看上面getEarlyBeanReference()这个方法所在的类,它是SpringAOP自动代理的关键类,它实现了SmartInstantiationAwareBeanPostProcessor,也就是说它也是个后置处理器BeanPostProcessor,它有着自定义的初始化后的方法。

/**
 * Create a proxy with the configured interceptors if the bean is
 * identified as one to proxy by the subclass.
 * @see #getAdvicesAndAdvisorsForBean
 */
@Override
public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) {
  if (bean != null) {
   Object cacheKey = getCacheKey(bean.getClass(), beanName);
   if (this.earlyProxyReferences.remove(cacheKey) != bean) {
     return wrapIfNecessary(bean, beanName, cacheKey);
   }
  }
  return bean;
}

很明显它这里是earlyProxyReferences缓存中找不到当前的bean的话就会去创建代理。也就是说SpringAOP希望在Bean初始化后进行创建代理。如果我们只使用二级缓存,也就是在这个地方

 addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));

直接调用getEarlyBeanReference()并将得到的早期引用放入二级缓存。这就意味着无论bean之间是否存在互相依赖,只要创建bean走到这一步都得去创建代理对象了。然而Spring并不想这么做,不信自己可以动手debug一下,如果ServiceA和ServiceB之间没有依赖关系的话,getEarlyBeanReference()这个方法压根就不会执行。总的来说就是,如果不使用三级缓存直接使用二级缓存的话,会导致所有的Bean在实例化后就要完成AOP代理,这是没有必要的。

最后我们重新梳理下流程,记得Spring创建Bean的时候是按照自然顺序的,所以A在前B在后:

我们首先进行A的创建,但由于依赖了B,所以开始创建B,同样的,对B进行属性注入的时候会要用到A,那么就会通过getBean()去获取A,A在实例化阶段会提前将对象放入三级缓存中,如果没有使用AOP,那么本质上就是这个bean本身,否则是AOP代理后的代理对象。三级缓存singletonFactories会将其存放进去。那么通过getBean()方法获取A的时候,核心其实在于getSingleton()方法, 它会将其从三级缓存中取出,然后放到二级缓存中去。而最终B创建结束回到A初始化的时候,会再次调用一次getSingleton()方法,此时入参的allowEarlyReference为false,因此是去二级缓存中取,得到真正需要的bean或代理对象,最后A创建结束,流程结束。

所以Spring解决循环依赖的原理大致就讲完了,但根据上述的结论,我们可以思考一个问题,什么情况的循环依赖是无法解决的?

根据上面的流程图,我们知道,要解决循环依赖首先一个大前提是bean必须是单例的,基于这个前提我们才值得继续讨论这个问题。然后根据上述总结,可以知道,每个bean都是要进行实例化的,也就是要执行构造器。所以能不能解决循环依赖问题其实跟依赖注入的方式有关。

依赖注入的方式有setter注入,构造器注入和Field方式。

Filed方式就是我们平时用的最多的,属性上加个@Autowired或者@Resource之类的注解,这个对解决循环依赖无影响;

如果A和B都是通过setter注入,显然对于执行构造器没有影响,所以不影响解决循环依赖;

如果A和B互相通过构造器注入,那么执行构造器的时候也就是实例化的时候,A在自己还没放入缓存的时候就去创建B了,那么B也是拿不到A的,因此会出错;

如果A中注入B的方式为setter,B中注入A为构造器,由于A先实例化,执行构造器,并创建缓存,都没有问题,继续属性注入,依赖了B然后走创建B的流程,获取A也可以从缓存里面能取到,流程一路通畅。

如果A中注入B的方式为构造器,B中注入A为setter,那么这个时候A先进入实例化方法,发现需要B,那么就会去创建B,而A还没放入三级缓存里,B再创建的时候去获取A就会获取失败。

好了,以上就是关于Spring解决循环依赖问题的所有内容,这个问题的答案我是很久之前就知道了,但真的只是知道答案,这次是自己看源码加debug一点点看才知道为啥是这个答案,虽然还做不到彻底学的通透,但的确能对这个问题的理解的更为深刻一点,再接再厉吧。

以上就是spring 如何解决循环依赖的详细内容,更多关于spring 循环依赖的资料请关注我们其它相关文章!

(0)

相关推荐

  • Spring IOC原理补充说明(循环依赖、Bean作用域等)

    前言 通过之前的几篇文章将Spring基于XML配置的IOC原理分析完成,但其中还有一些比较重要的细节没有分析总结,比如循环依赖的解决.作用域的实现原理.BeanPostProcessor的执行时机以及SpringBoot零配置实现原理(@ComponentScan.@Import.@ImportSource.@Bean注解的使用和解析)等等.下面就先来看看循环依赖是怎么解决的,在此之前一定要熟悉整个Bean的实例化过程,本篇只会贴出关键性代码. 正文 循环依赖 首先来看几个问题: 什么是循环依

  • Spring为何要用三级缓存来解决循环依赖问题

    我们都知道Spring为了解决循环依赖使用了三级缓存 Spring三级缓存 一级缓存singletonObjects 用于保存BeanName和创建bean实例之间的关系,beanName -> bean instance private final Map<String, Object> singletonObjects = new ConcurrentHashMap(256); 二级缓存earlySingletonObjects 保存提前曝光的单例bean对象 private fin

  • Spring循环依赖的解决办法,你真的懂了吗

    介绍 先说一下什么是循环依赖,循坏依赖即循环引用,两个或多个bean相互引用,最终形成一个环.Spring在初始化A的时候需要注入B,而初始化B的时候需要注入A,在Spring启动后这2个Bean都要被初始化完成 Spring的循环依赖有两种场景 构造器的循环依赖 属性的循环依赖 构造器的循环依赖,可以在构造函数中使用@Lazy注解延迟加载.在注入依赖时,先注入代理对象,当首次使用时再创建对象完成注入 属性的循环依赖主要是通过3个map来解决的 构造器的循环依赖 @Component publi

  • 详解Spring循环依赖的解决方案

    spring针对Bean之间的循环依赖,有自己的处理方案.关键点就是三级缓存.当然这种方案不能解决所有的问题,他只能解决Bean单例模式下非构造函数的循环依赖. 我们就从A->B->C-A这个初始化顺序,也就是A的Bean中需要B的实例,B的Bean中需要C的实例,C的Bean中需要A的实例,当然这种需要不是构造函数那种依赖.前提条件有了,我们就可以开始了.毫无疑问,我们会先初始化A.初始化的方法是org.springframework.beans.factory.support.Abstra

  • 浅谈Spring 解决循环依赖必须要三级缓存吗

    我们都知道 Spring 是通过三级缓存来解决循环依赖的,但是解决循环依赖真的需要使用到三级缓冲吗?只使用两级缓存是否可以呢?本篇文章就 Spring 是如何使用三级缓存解决循环依赖作为引子,验证两级缓存是否可以解决循环依赖. 循环依赖 既然要解决循环依赖,那么就要知道循环依赖是什么.如下图所示: 通过上图,我们可以看出: A 依赖于 B B 依赖于 C C 依赖于 A public class A { private B b; } public class B { private C c; }

  • spring解决循环依赖的简单方法

    Spring内部如何解决循环依赖,一定是单默认的单例Bean中,属性互相引用的场景.比如几个Bean之间的互相引用: 或者 setter方式原型,prototype 原型(Prototype)的场景是不支持循环依赖的,因为"prototype"作用域的Bean,为每一个bean请求提供一个实例,Spring容器不进行缓存,因此无法提前暴露一个创建中的Bean,会抛出异常. 构造器参数循环依赖 Spring容器会将每一个正在创建的Bean 标识符放在一个"当前创建Bean池&q

  • spring解决循环依赖

    概述 循环依赖就是依赖关系形成环,比如最简单的循环依赖:A对象依赖B,B对象依赖A 属性注入与循环依赖 如果是构造器注入,如果循环依赖对象没法构建,因为还未实例化 如果是属性注入但是作用域是prototype,spring不会缓存其对象实例,也不能处理循环依赖的情况 如果是属性注入singleton的,其bean的实例化过程与属性注入过程是分开的,并且spring提供了三个map(就是大家说三级缓存)来实现. spring属性注入处理循环依赖的方式 通过以下xml方式配置一个循环依赖的示例: <

  • Spring如何解决循环依赖的问题

    前言 在面试的时候这两年有一个非常高频的关于spring的问题,那就是spring是如何解决循环依赖的.这个问题听着就是轻描淡写的一句话,其实考察的内容还是非常多的,主要还是考察的应聘者有没有研究过spring的源码.但是说实话,spring的源码其实非常复杂的,研究起来并不是个简单的事情,所以我们此篇文章只是为了解释清楚Spring是如何解决循环依赖的这个问题. 什么样的依赖算是循环依赖? 用过Spring框架的人都对依赖注入这个词不陌生,一个Java类A中存在一个属性是类B的一个对象,那么我

  • 简单了解Spring beanfactory循环依赖命名重复2大属性

    是否允许循环依赖和bean的命名重复取决于beanfactory的两大属性 allowBeanDefinitionOverriding和allowCircularReferences.这两个属性未指定时取的是DefaultListableBeanFactory中的默认值,均为true,即默认允许同名bean和循环引用(只有单例允许). 但是假如读者用的springboot版本比较新,就会发现情况并不是这样,同名bean会冲突报错.为什么呢,跟踪源码发现是Springboot应用在2.1.8版本的

  • 浅谈Spring如何解决循环依赖的问题

    在关于Spring的面试中,我们经常会被问到一个问题,就是Spring是如何解决循环依赖的问题的.这个问题算是关于Spring的一个高频面试题,因为如果不刻意研读,相信即使读过源码,面试者也不一定能够一下子思考出个中奥秘.本文主要针对这个问题,从源码的角度对其实现原理进行讲解. 1. 过程演示 关于Spring bean的创建,其本质上还是一个对象的创建,既然是对象,读者朋友一定要明白一点就是,一个完整的对象包含两部分:当前对象实例化和对象属性的实例化.在Spring中,对象的实例化是通过反射实

  • 基于SpringBoot构造器注入循环依赖及解决方式

    1. 循环依赖是什么? Bean A 依赖 B,Bean B 依赖 A这种情况下出现循环依赖. Bean A → Bean B → Bean A 更复杂的间接依赖造成的循环依赖如下. Bean A → Bean B → Bean C → Bean D → Bean E → Bean A 2. 循环依赖会产生什么结果? 当Spring正在加载所有Bean时,Spring尝试以能正常创建Bean的顺序去创建Bean. 例如,有如下依赖: Bean A → Bean B → Bean C Spring

随机推荐