详解Spring Bean的循环依赖解决方案
如果使用构造函数注入,则可能会创建一个无法解析的循环依赖场景。
什么是循环依赖
循环依赖其实就是循环引用,也就是两个或则两个以上的bean互相持有对方,最终形成闭环。比如a依赖于b,b依赖于c,c又依赖于a。如下图:
注意,这里不是函数的循环调用,是对象的相互依赖关系。循环调用其实就是一个死循环,除非有终结条件。
spring中循环依赖场景有:
(1)构造器的循环依赖
(2)field属性的循环依赖。
怎么检测是否存在循环依赖
检测循环依赖相对比较容易,bean在创建的时候可以给该bean打标,如果递归调用回来发现正在创建中的话,即说明了循环依赖了。
下面是我所遇到的情况,代码结构如下:
springsecurity 配置类:
@configuration public class browsersecurityconfig extends websecurityconfigureradapter { private final userdetailsservice userdetailsservice; /** * 通过配置类构造函数注入 userdetailsservice */ @autowired public browsersecurityconfig(userdetailsservice userdetailsservice) { this.userdetailsservice = userdetailsservice; } /** * 在配置类中声明 加密编码器 */ @bean public passwordencoder passwordencoder() { return new bcryptpasswordencoder(); } ... ... }
userdetailsservice 类:
@component public class myuserdetailservice implements userdetailsservice { private final passwordencoder passwordencoder; private logger logger = loggerfactory.getlogger(getclass()); /** * 通过构造函数注入 passwordencoder */ @autowired public myuserdetailservice(passwordencoder passwordencoder) { this.passwordencoder = passwordencoder; } ... ... }
运行之后,spring抛出了如下错误信息:
description:
the dependencies of some of the beans in the application context form a cycle:
┌─────┐
| browsersecurityconfig defined in file [d:\code\java\ideaprojects\mango-security\mango-security-browser\target\classes\stu\mango\security\browser\browsersecurityconfig.class]
↑ ↓
| myuserdetailservice defined in file [d:\code\java\ideaprojects\mango-security\mango-security-browser\target\classes\stu\mango\security\browser\myuserdetailservice.class]
└─────┘
该例中,browsersecurityconfig 通过构造函数注入 userdetailsservice实例,而 userdetailsservice由通过构造函数注入在browsersecurityconfig 中声明的passwordencoder。
总结来说,spring bean的循环依赖是指,类a需要通过构造函数注入的类b的实例(或者b中声明的bean),而类b需要通过构造函数注入的类a的实例(或者a中声明的bean)。如果将类a和类b的bean配置为相互注入,则spring ioc容器会在运行时检测到此循环引用,并引发一个beancurrentlyincreationexception。与典型情况(没有循环依赖)不同,bean a和bean b之间的循环依赖关系迫使其中一个bean在被完全初始化之前被注入到另一个bean中(这是一个典型的“先有鸡还是先有蛋”场景)。
解决方案
简明扼要的说,就是——不使用基于构造函数的依赖注入。可通过下面方式解决。
在字段上使用@autowired注解,让spring决定在合适的时机注入。【推荐】
用基于setter方法的依赖注射取代基于构造函数的依赖注入来解决循环依赖。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。