在使用了lombok注解 @RequiredArgsConstructor 注入服务时,启动项目报了循环依赖的问题
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v2.0.4.RELEASE)
2021-11-29 11:46:12.427||main||ERROR||TID: N/A||org.springframework.boot.diagnostics.LoggingFailureAnalysisReporter||
***************************
APPLICATION FAILED TO START
***************************
Description:
The dependencies of some of the beans in the application context form a cycle:
thirdPreReviewController defined in URL [jar:file:/service-manager-rest-1.0-SNAPSHOT.jar!/BOOT-INF/classes!/com/service/manager/rest/ThirdPreReviewController.class]
┌─────┐
| policyServiceImpl defined in URL [jar:file:/service-manager-rest-1.0-SNAPSHOT.jar!/BOOT-INF/lib/ins-service-manager-service-1.0-SNAPSHOT.jar!/com/service/manager/service/PolicyServiceImpl.class]
↑ ↓
| serviceManagerServiceImpl defined in URL [jar:file:/service-manager-rest-1.0-SNAPSHOT.jar!/BOOT-INF/lib/service-manager-service-1.0-SNAPSHOT.jar!/com/service/manager/service/service/ServiceManagerServiceImpl.class]
└─────┘
先看下注解RequiredArgsConstructor的使用和作用
1.必须声明的变量为final
2.根据构造器注入的,相当于当容器调用带有一组参数的类构造函数时,基于构造函数的 DI
就完成了,其中每个参数代表一个对其他类的依赖。基于构造方法为属性赋值,容器通过调用类的构造方法将其进行依赖注入
使用方式就是在服务类上面加上此注解,然后使用其他服务的时候直接引入即可,要求必须使用final修饰
因为根据构造方法注入 所以发生了循环依赖
总结来说,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方法的依赖注射取代基于构造函数的依赖注入来解决循环依赖。