spring源码分析 循环依赖

首先出现循环依赖肯定是自己的代码设计出现了问题,但是保不准真有这种情况发生啊。
spring对原型模式的循环依赖的处理很简单,直接就抛出异常。创建失败

protected <T> T doGetBean(
            final String name, final Class<T> requiredType, final Object[] args, boolean typeCheckOnly)
            throws BeansException {

        ......
            if (isPrototypeCurrentlyInCreation(beanName)) {
                throw new BeanCurrentlyInCreationException(beanName);
            }
        ......

下面讲一下spring能够处理循环依赖的注入方式 setter 方式。

先看一下简单例子

public class ServiceA {
    private String name;

    private ServiceB serviceB;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public ServiceB getServiceB() {
        return serviceB;
    }

    public void setServiceB(ServiceB serviceB) {
        this.serviceB = serviceB;
    }
}


public class ServiceA {
    private String name;
    private ServiceB serviceB;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public ServiceB getServiceB() {
        return serviceB;
    }

    public void setServiceB(ServiceB serviceB) {
        this.serviceB = serviceB;
    }
}


public class ServiceA {
    private String name;
    private ServiceB serviceB;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public ServiceB getServiceB() {
        return serviceB;
    }

    public void setServiceB(ServiceB serviceB) {
        this.serviceB = serviceB;
    }
}

/**
 * 测试类
 * @author hepei
 * @date 2017年9月24日 下午11:37:43
 */
public class Test {
    @SuppressWarnings("deprecation")
    public static void main(String[] args) {
        XmlBeanFactory f=new XmlBeanFactory(new ClassPathResource("spring-context.xml"));
        ServiceA serviceA=(ServiceA) f.getBean("serviceA");
        System.out.println(serviceA.getName());
        System.out.println(serviceA.getServiceB().getServiceC().getPhone());
    }

}

spring-context.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
      http://www.springframework.org/schema/beans/spring-beans-4.0.xsd">

    <bean id="serviceA" class="com.eroadsf.springdemo.ServiceA">
        <property name="name" value="何"></property>
        <property name="serviceB" ref="serviceB"></property>
    </bean>

    <bean id="serviceB" class=" com.eroadsf.springdemo.ServiceB">
        <property name="address" value="玄武"></property>
        <property name="serviceC" ref="serviceC"></property>
    </bean>
    <bean id="serviceC" class="com.eroadsf.springdemo.ServiceC">
        <property name="phone" value="138"></property>
        <property name="serviceA" ref="serviceA"></property>
    </bean>
</beans>

运行结果

138

下面通过源码分析一下spring是如何处理setter方式的循环依赖的。

这里写图片描述

这里写图片描述

这里写图片描述

这里写图片描述

Spring通过三级缓存加上“提前曝光”机制,配合Java的对象引用原理,比较完美地解决了某些情况下的循环依赖问题!

这里写图片描述

单例在Spring的同一个容器内只会被创建一次,后续在获取bean,就直接从单例缓存中获取。首先尝试从缓存中加载,如果加载不成功的话再尝试从singletonFactories中加载。因为在创建单例bean的时候会存在依赖注入的情况,而在创建依赖的时候为了避免循环依赖,在spring中创建bean的原则是不等bean创建完成就会将创建bean的ObjectFactory提早曝光加入到缓存中,一旦下一个bean创建时候需要依赖上一个bean则直接使用ObjectFactory.

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring循环依赖是指两个或多个Bean之间相互依赖,形成一个循环引用的关系。在Spring容器启动的过程中,如果存在循环依赖,那么默认情况下会抛出BeanCurrentlyInCreationException异常。 下面是Spring循环依赖源码分析: 1. 当我们向Spring容器中注入一个Bean时,Spring会先检查这个Bean是否已经正在创建中(正在创建的Bean是无法注入的),如果正在创建中,则直接返回一个早期引用,否则继续创建Bean。 2. 在Bean的创建过程中,当遇到依赖注入(如@Autowired注解)时,Spring会检查要注入的Bean是否已经在创建中。如果是,则返回一个代理对象作为占位符,等待真正的Bean创建完毕后再进行注入。 3. Spring使用三级缓存来解决循环依赖问题。第一级缓存是单例池,存放已经创建好的单例Bean。第二级缓存是提前暴露的ObjectFactory,存放早期引用。第三级缓存是用于存放正在创建中的Bean的缓存,用于检测循环依赖。 4. 当检测到循环依赖时,Spring会尝试使用构造函数的方式完成循环依赖。它会先创建一个空对象,并将其放入到第三级缓存中。然后调用构造函数去创建这个Bean,此时依赖的Bean会返回一个早期引用。最后,将这个Bean加入到第一级缓存中,并开始注入依赖。 5. 当所有的Bean都创建完成后,Spring会触发后置处理器的回调方法,完成Bean的初始化。 总结:Spring循环依赖通过使用三级缓存和构造函数来解决,在Bean创建过程中动态地判断和处理循环依赖关系,确保所有的Bean都能被正确地创建和注入。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值