spring循环依赖问题深究

一、背景

清分服务添加一个异步处理功能(@asyc),本地测试时发现启动服务后有时正常有时异常。

问题分析

1、相同的环境,启动服务结果不同,定位为工程代码引起该异常。

2、启动类中,配置文件名采用了通配符,理论上配置文件加载顺序不固定。

3、从机器上的日志可以看出,启动成功和异常时spring/application-xxx.xml加载顺序不同。

4、从错误日志可以看出两个问题点:循环依赖和不同的bean依赖了同一个bean的不同版本。(spring默认支持循环依赖)

说明:

1、bean的实例化过程:create bean →  cache bean → inject element → inject element → inject element ... ... → finish(debug模式可以跟踪完整的bean实例化过程)

2、非循环依赖场景

条件:A依赖B,如果先create A

实例化过程为:create A → cache A → create B → cache B → finish B → inject B to A → finish A。(被依赖对象先finish,即B先于A实例化完成)

3、循环依赖场景

条件:A依赖B,B依赖A,如果先create A

实例化过程为:create A → cache A → create B → cache B → inject A to B → finish B → inject B to A → finish A。(spring允许eagerly cached instance注入到其他bean中,eagerly cached instance 指已缓存但未finish的bean)

清分工程中bean的依赖比较复杂,抽象出下面的场景做说明:(下面的bean都为单例模式,非单例模式下bean不会进行cache,不支持循环依赖)

步骤1:A依赖B、C依赖B,B依赖C,并且B实现类的方法上存在@async注解,即B在实例化完成前会生成代理。
步骤2:A开始create,发现B未实例化,于是先cache A。
步骤3:B开始create,发现C未实例化,于是先cache B。
步骤4:C开始create,发现B已实例化,取出cache的B,注入C中,C finish。
(为什么B未实例化完就注入到C中了?that is not fully initialized yet - a consequence of a circular reference,因为存在循环依赖,所以允许先注入。猜测未做是否被代理的检测)
步骤5:生成B的代理类[email protected]。
步骤6:[email protected]注入到A时,发现[email protected]的原始实例B已注入到循环引用的C中,却没有使用B的最终代理类[email protected],所以抛异常。

正常情况应该为B先finish,A和C后finish,这样可以保证B的最终代理类注入到A和C中。

小心陷阱

1、depends-on设置:如果A depends-on B,只会保证B先于A开始实例化(Creating instance of bean),但不保证B先于A实例完成(Finished creating instance of bean)。对于简单的bean,实例化完成顺序正常。

2、尽量保持bean的依赖关系简单。

3、保证bean加载顺序固定。

4、service注入的时候尽量注入mapper,否则经过多次迭代业务更改,循环依赖就容易暴露出来。
————————————————
版权声明:本文为CSDN博主「白底黑字」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/lianhuazy167/article/details/66967673

原文地址:https://www.cnblogs.com/yoishion/p/11595140.html

时间: 2024-09-30 07:16:42

spring循环依赖问题深究的相关文章

spring循环依赖问题分析——待后续补充

新搞了一个单点登录的项目,用的cas,要把源码的cas-webapp改造成适合我们业务场景的项目,于是新加了一些spring的配置文件. 但是在项目启动时报错了,错误日志如下: 一月 08, 2016 11:01:34 下午 org.apache.catalina.core.StandardContext filterStart 严重: Exception starting filter springSecurityFilterChain org.springframework.beans.fa

SPRING循环依赖(circular reference)的解决方法

循环依赖,就是说类A依赖与B,而B又依赖于A,这种情况本不应该发生,发生这种情况是因为我在项目中使用的工厂模式,用一个工厂来生产一些管理器类,而某一管理器要需要另一管理器提供支持所以就要引用工厂类,而这个管理器和这个工厂就出现了循环依赖(项目中实际的逻辑比这个更复杂,因为我在项目中实现的了一个工作流数据POJO类延迟加载的功能像hibernate 那样在调用一个类的集合属性时才到要shark中去查找数据而不是在new里加载,并且这个数据类的集合属性并不包含加载数据的代码只是普通的Bean方法ge

Spring循环依赖

Spring-bean的循环依赖以及解决方式 Spring里面Bean的生命周期和循环依赖问题 什么是循环依赖? 循环依赖其实就是循环引用,也就是两个或者两个以上的bean互相持有对方,最终形成闭环.比如A依赖于B,B依赖于C,C又依赖于A.如下图: 注意,这里不是函数的循环调用,是对象的相互依赖关系.循环调用其实就是一个死循环,除非有终结条件. Spring中循环依赖场景有: (1)构造器的循环依赖 (2)field属性的循环依赖 其中,构造器的循环依赖问题无法解决,只能拋出BeanCurre

面试中被问Spring循环依赖的三种方式!!!

什么是循环依赖? 循环依赖其实就是循环引用,也就是两个或则两个以上的 Bean 互相持有对方,最终形成闭环.比如A依赖于B,B依赖于C,C又依赖于A.如下图: 如果在日常开发中我们用new 对象的方式发生这种循环依赖的话程序会在运行时一直循环调用,直至内存溢出报错.下面说一下Spring是如果解决循环依赖的. 注意,这里不是函数的循环调用,是对象的相互依赖关系.循环调用其实就是一个死循环,除非有终结条件. 根据 spring 中 Bean 的注入方式:构造器注入方式,属性注入方式(单例和多例),

spring循环依赖是怎么解决的?

回答:循环依赖就是循环引用,就是两个或多个Bean相互之间的持有对方,比如CircleA引用CircleB,CircleB引用CircleA,则它们最终反映为一个环. Spring如何解决循环依赖? 假设场景如下,A->B->A 1.实例化A,并将未注入属性的A暴露出去,即提前曝光给容器Wrap2.开始为A注入属性,发现需要B,调用getBean(B)3.实例化B,并注入属性,发现需要A的时候,从单例缓存中查找,没找到时继而从Wrap中查找,从而完成属性的注入4.递归完毕之后回到A的实例化过程

Spring循环依赖原因及如何解决

浅谈Spring解决循环依赖的三种方式 SpringBoot构造器注入循环依赖及解决 原文:https://www.baeldung.com/circular-dependencies-in-spring 代码:https://github.com/eugenp/tutorials/tree/master/spring-di 原文地址:https://www.cnblogs.com/liran123/p/11655604.html

01 spring循环依赖

作者:Mythsman原文:https://blog.mythsman.com/post/5d838c7c2db8a452e9b7082c/转载自:猿天地原文来自:https://mp.weixin.qq.com/s/pIawXWHCGqIHi0INHDgawQ 1 什么是依赖 其实,不分场景地.笼统地说A依赖B其实是不够准确.至少是不够细致的.我们可以简单定义一下什么是依赖. 所谓A依赖B,可以理解为A中某些功能的实现是需要调用B中的其他功能配合实现的.这里也可以拆分为两层含义: A强依赖B.

Spring中-IOC-Bean的初始化-循环依赖的解决

前言 在实际工作中,经常由于设计不佳或者各种因素,导致类之间相互依赖.这些类可能单独使用时不会出问题,但是在使用Spring进行管理的时候可能就会抛出BeanCurrentlyInCreationException等异常 .当抛出这种异常时表示Spring解决不了该循环依赖,本文将简要说明Spring对于循环依赖的解决方法. 循环依赖的产生和解决的前提 循环依赖的产生可能有很多种情况,例如: A的构造方法中依赖了B的实例对象,同时B的构造方法中依赖了A的实例对象 A的构造方法中依赖了B的实例对象

Spring之循环依赖

转:http://my.oschina.net/tryUcatchUfinallyU/blog/287936 概述 如何检测循环依赖 循环依赖如何解决 Spring如何解决循环依赖 主要的几个缓存 主要步骤 Spring不能完全解决的循环依赖问题 面对Spring不能完全解决的现状,我们该如何处理 概述 入职的时候学习spring研究过循环依赖,现在再回顾下,发现啥都忘记了,还是得总结下来,故总结该文. 本文主要解决如下问题: 1.何为循环依赖 2.如何检测循环依赖 3.循环依赖可以如何解决 4