SprinBoot 如何解决循环调用

SprinBoot 如何解决循环调用

此文仅记录自身遇到的问题进行解决处理思路。
背景: 项目为springboot-1.5的版本,迁移上阿里云,提升springboot版本为2.6.当初最开始的时候用的是jetty容器在本地进行启动,但是最后需要将工程迁移到ali cloud,并且运用K8S作为容器,所以最后将项目打包方式改为jar包。Jetty 运行的时候是没有丝毫的问题,但是用自带的tomcat 的时候问题就暴漏出来了。
原因应该是循环引用对于springboot2.6来说是不被鼓励的,所以会直接暴漏出来,最直接简单粗暴的方法:

第一种解决思路:
下面为允许循环调用的配置操作:

spring.main.allow-circular-references=true

第二种解决思路:
因为用最简单的方式进行循环调用,表象一是自己故意做出来的,表象2为当时遇到的。故将两种表象都进行记录,
所谓循环调用无非就是A调用B,B调用A,导致的springboot 在进行对象加载的时候的一种冲突。下面具体介绍解决思路.
表象1:此为最简单,也是最容易发现的一种循环调用表现:

2022-10-02 08:44:55.456  WARN 17112 --- [           main] s.c.a.AnnotationConfigApplicationContext : Exception encountered during context initialization - cancelling refresh attempt: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'a': Unsatisfied dependency expressed through field 'b'; nested exception is org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'b': Unsatisfied dependency expressed through field 'a'; nested exception is org.springframework.beans.factory.BeanCurrentlyInCreationException: Error creating bean with name 'a': Requested bean is currently in creation: Is there an unresolvable circular reference?
2022-10-02 08:44:55.456  INFO 17112 --- [           main] ConditionEvaluationReportLoggingListener : 

Error starting ApplicationContext. To display the conditions report re-run your application with 'debug' enabled.
2022-10-02 08:44:55.472 ERROR 17112 --- [           main] o.s.b.d.LoggingFailureAnalysisReporter   : 

***************************
APPLICATION FAILED TO START
***************************

Description:

The dependencies of some of the beans in the application context form a cycle:

┌─────┐
|  a (field private com.example.service.B com.example.service.A.b)
↑     ↓
|  b (field private com.example.service.A com.example.service.B.a)
└─────┘

表象2: 此种直接导致tomcat启动失败,并且在关闭tomcat的时候直接失败,且出现了内存泄漏的问题,错误是一样的,顺手从网上粘贴了一个

 07-Sep-2020 19:09:11.196 WARNING [localhost-startStop-1] org.apache.catalina.loader.WebappClassLoaderBase.clearReferencesThreads The web application [system] appears to have started a thread named [pool-1-thread-4] but has failed to stop it. This is very likely to create a memory leak. Stack trace of thread:
 sun.misc.Unsafe.park(Native Method)
 java.util.concurrent.locks.LockSupport.park(LockSupport.java:175)
 java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(AbstractQueuedSynchronizer.java:2039)
 java.util.concurrent.LinkedBlockingQueue.take(LinkedBlockingQueue.java:442)
 java.util.concurrent.ThreadPoolExecutor.getTask(ThreadPoolExecutor.java:1074)
 java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1134)
 java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
 java.lang.Thread.run(Thread.java:748)

表象一直接是在控制台看到的,没啥好说的
表象二这种应用启动的时候直接就提示会造成内存泄漏,但是具体的错误信息是没有的。

当时解决的时候是分为了一下几步:
1.main方法进行debug模式。调节logbcak日志级别为debug,但是效果不是很明显。依然看不到具体的错误信息。
2.然后自己就只能盲猜了,直接上断点给各种config注解,datasource,很幸运,当加载datasource 的时候报错了。然后放开断点,没问题出现了表象2的问题,原因是我们使用的druid连接池里面的filter配置,最后注释掉,加载成功了,但是问题依然存在。

那就干脆直接从断点出入手,一路加载debug下去,最后发现了create bean 的时候出现了问题,此时便发现了循环调用的问题。
OK,处理掉之后,本地已经可以正常run 起来了。然而,在上到服务器的时候依旧报表象2的错误。
3. 既然已经知道是循环调用的问题,那么网上搜个logback的配置吧。让它直接打印各个对象的加载和调用log吧

配它!!

<logger name="org.springframework.boot">
	<level value="trace"/>
	<appender-ref ref="stdout"/>
</logger>
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 在Spring Boot中,循环依赖指的是两个或多个Bean之间出现的相互依赖关系,而导致无法正确地初始化Bean的情况。Spring Boot中提供了以下几种方式来解决循环依赖的配置问题: 1. 推荐使用构造函数注入:在Bean的构造函数中接收依赖的Bean,将依赖作为参数传递进去,并标注`@Autowired`注解。这种方式可以确保Bean在创建过程中所有的依赖都已经被注入进来,避免了循环依赖的问题。 2. 使用`@Lazy`注解延迟加载@Bean:将存在循环依赖的Bean标记为`@Lazy`,这样Spring容器会在需要时才去创建该Bean,避免了循环依赖的问题。 3. 使用`@PostConstruct`注解进行初始化:在存在循环依赖的Bean中,可以使用`@PostConstruct`注解标注一个初始化方法,在该方法中进行对依赖的操作。这样,当依赖的Bean创建完成后,初始化方法会被自动调用解决循环依赖的问题。 4. 使用Setter注入:在Bean中使用Setter方法注入依赖,将依赖作为参数传递给Setter方法,并标注`@Autowired`注解。这种方式同样可以避免循环依赖的问题。 需要注意的是,虽然Spring Boot提供了上述解决循环依赖的方式,但尽量还是要避免出现循环依赖的情况。当出现循环依赖时,需要仔细检查代码逻辑,考虑是否可以重构代码或者优化依赖关系,以减少依赖的嵌套层次,降低耦合度。 ### 回答2: 在Spring Boot解决循环依赖可以通过以下几种方式来配置: 1. 构造函数注入:在类的构造函数中通过参数注入依赖对象。这样可以确保依赖对象在实例化时已经存在,避免了循环依赖的问题。 2. @Autowired注解:使用@Autowired注解来标注需要注入的对象。Spring Boot会自动处理依赖的注入和解决循环依赖的问题。需要注意的是,如果存在循环依赖,Spring Boot会抛出BeanCurrentlyInCreationException异常,提示出现循环依赖的情况。 3. @Lazy注解:在需要注入的对象上添加@Lazy注解,延迟加载依赖对象。这样可以通过在需要使用依赖对象的地方进行注入,避免了循环依赖的问题。 4. Setter方法注入:通过Setter方法来注入依赖对象。这样可以确保依赖对象在实例化后再进行注入,避免了循环依赖的问题。 以上是解决循环依赖的一些常用配置方式。根据具体的场景和需求,可以选择适合的方式来解决循环依赖的问题。在配置的同时,需要谨慎处理循环依赖可能引发的性能问题和死锁等情况,确保应用程序的正常运行。 ### 回答3: 在SpringBoot中,如果存在循环依赖的配置,可以通过以下方式解决: 1. 使用构造函数注入:将循环依赖的对象作为构造函数的参数,将其注入到另一个对象中。这样在创建对象时,就能够解决循环依赖的问题。 2. 使用@Lazy注解:通过在需要延迟加载的类上添加@Lazy注解,可以解决循环依赖的问题。@Lazy注解表示在第一次使用时才进行初始化,避免了循环依赖的影响。 3. 使用@DependsOn注解:通过在需要先初始化的类上添加@DependsOn注解,指定依赖的顺序,可以解决循环依赖的问题。这样在创建Bean时,会按照指定的依赖顺序进行初始化,避免了循环依赖的错误。 4. 使用setter方法注入:将循环依赖的对象注入到另一个对象中的setter方法中。在创建对象后,通过setter方法将所依赖的对象注入进去,解决循环依赖的问题。 总的来说,SpringBoot提供了多种解决循环依赖的方式,可以根据具体的场景选择合适的方式来解决问题。但需要注意的是,循环依赖的存在可能会导致性能下降和程序的复杂性增加,因此在设计时应尽量避免循环依赖的发生。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值