构造函数注入,为什么不能解决循环依赖

文章探讨了解决循环依赖的关键在于将实例化和初始化分开。构造函数注入在解析参数时创建对象,可能导致循环依赖问题,而属性注入则在对象已实例化后进行,避免了该问题。通过XML配置示例展示了Spring如何处理这两种注入方式的不同处理机制。
摘要由CSDN通过智能技术生成

1. 结论

解决循环依赖的核心是,实例化和初始化分开

构造函数注入和属性注入,如果是引用类型,那么创建引用对象的时机是不同的。

  • 对于构造函数注入,是在解析构造函数参数的时候去创建引用对象的,此时当前的类还没有完成实例化,如果相互引用那么实例化对象都不能完成的话,就无法解决循环依赖。
autowireConstructor(){
//.....
minNrOfArgs = resolveConstructorArguments(beanName, mbd, bw, cargs, resolvedValues);
}
  • 属性注入是在填充属性的时候去创建对象的,此时当前的类型已经完成的实例化

2.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"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">

<!--隔离==============================属性====================================================隔离-->
<!-- 自动装配,根据属性,将对应的对象,自动装配进去   -->
<!--    这种配置,会走到populateBean()的autowireByName()方法中;是对象,那么继续走实例化的操作-->
    <bean id="person" class="com.songbl.testinject.Person" autowire="byName" >
        <property name="id" value="1"></property>
        <property name="name" value="zhangsan"></property>
<!--        自动装配的话,就省掉了这行操作-->
<!--        <property name="address" ref="address"></property>-->
        <!-- 其中如果是构造函数注入的话,那么在解析参数的时候,就实例化对象的操作了-->
<!--        <constructor-arg name="id" value="1"></constructor-arg>-->
<!--        <constructor-arg name="address" ref="address"></constructor-arg>-->
<!--        <constructor-arg name="name" value="lisi"></constructor-arg>-->
    </bean>
<!--    即使上面是的配置,但是的autowireByName里面containsBean(propertyName),不加上这个还是会报错;为什么这么写呢,其实就是省掉了ref-->
    <bean id="address" class="com.songbl.testinject.Address"/>
<!--隔离==============================属性================================================隔离-->




<!--隔离=============================构造函数============================================隔离-->
<!-- 构造函数注入的,在解析构造参数的时候,需要的对象,就实例化了。ConstructorResolver的resolveConstructorArguments -->
<bean id="person" class="com.songbl.testinject.Person" >
    <!-- 其中如果是构造函数注入的话,那么在解析参数的时候,就实例化对象的操作了-->
            <constructor-arg name="id" value="1"></constructor-arg>
            <constructor-arg name="name" value="lisi"></constructor-arg>
            <!--  指向Address的bean配置  -->
           <constructor-arg name="address" ref="address"></constructor-arg>
</bean>
<!--隔离=====================================================================================隔离-->


<!--    构造函数注入和属性注入,如果是引用类型,那么创建引用对象的时机是不同的,对于构造函数注入,是在解析构造函数参数的时候去创建引用对象的,此时当前的类还没有完成实例化;
而属性注入是在填充属性的时候去创建对象的,此时当前的类型已经完成的实例化-->



</beans>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值