前言
实际开发中,咱们经常是基于模块分工开发的,也就是不一样的人负责不一样的模块。最后合并代码。这种方式适合多人协同,每一个人只关心本身的业务模块实现(controller/model/service/mapper等),当碰到须要其它模块支持的功能时,只需引入其它模块的类便可调用其方法html
循环依赖问题
Bean A 依赖 B,Bean B 依赖 A这种状况下出现循环依赖。
Bean A → Bean B → Bean Aapp
更复杂的间接依赖形成的循环依赖以下。
Bean A → Bean B → Bean C → Bean D → Bean E → Bean Athis
循环依赖问题的本质
当程序启动,Spring Context加载全部的Bean时,会尝试按他们运行的工做顺序建立Bean。.net
例如,有以下依赖:
Bean A → Bean B → Bean C
Spring先建立beanC,接着建立bean B(将C注入B中),最后建立bean A(将B注入A中)。设计
但当存在循环依赖时,Spring将没法决定先建立哪一个bean。这种状况下,Spring将产生异常BeanCurrentlyInCreationException。代理
举例
有一个ServiceA须要调用ServiceB的方法,那么ServiceA就依赖于ServiceB,那在ServiceB中再调用ServiceA的方法,就造成了循环依赖。Spring在初始化bean的时候就不知道先初始化哪一个,bean就会报错。code
解决
从新设计
从新设计结构,消除循环依赖。htm
使用注解 @Lazy
一种最简单的消除循环依赖的方式是经过延迟加载。在注入依赖时,先注入代理对象,当首次使用时再建立对象完成注入。
对象
@Autowired @Lazy private ServiceA serviceA; @Autowired @Lazy private ServiceB serviceB;
使用Setter/Field注入
Spring文档建议的一种方式是使用setter注入。当依赖最终被使用时才进行注入。blog
@Component public class ServiceA { private ServiceB serviceB; @Autowired public void setServiceB(ServiceB serviceB) { this.serviceB = serviceB; } public ServiceB getServiceB() { return serviceB; } }
使用@PostConstruct
@Component public class ServiceA { @Autowired private ServiceB serviceB; @PostConstruct public void init() { serviceB.setServiceA(this); } public ServiceB getServiceB() { return serviceB; } }