Circular dependencies in Spring

In this post, we will discuss one of the most asked Spring interview questions. What happens if there are circular dependencies in Spring.

Problem

What if there are circular dependencies in Spring?
For example: Class A requires an instance of Class B and Class B requires an instance of Class A.

Let’s create an example for the same.
1. Create a simple java maven project.
2. Maven dependency
put spring and cglib maven dependency in pom.xml.

So your pom.xml will look like:

3. Create Bean class

Create a bean class called A.java in package org.arpit.java2blog .
Create another bean class called B.java in package org.arpit.java2blog. This class will have dependency to above A.java .

4. ApplicationContext.xml
Create ApplicationContext.xml in src/main/resources as below.

Please note that we are using Contructor injection to inject dependencies here.
5. Create CircularDependencyMain.java

6. Run it
When you run above program, you will get below output.

Jun 15, 2018 11:21:58 AM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@31cefde0: startup date [Fri Jun 15 11:21:58 IST 2018]; root of context hierarchy
Jun 15, 2018 11:21:58 AM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [ApplicationContext.xml] Exception in thread “main” org.springframework.beans.factory.BeanCreationException: Error creating bean with name ‘a’ defined in class path resource [ApplicationContext.xml]: Cannot resolve reference to bean ‘b’ while setting constructor argument; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name ‘b’ defined in class path resource [ApplicationContext.xml]: Cannot resolve reference to bean ‘a’ while setting constructor argument; 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?
.
.
.
Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name ‘b’ defined in class path resource [ApplicationContext.xml]: Cannot resolve reference to bean ‘a’ while setting constructor argument; 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?
.
.
.

As you can see, we are getting BeanCurrentlyInCreationException.Requested bean is currently in creation: Is there an unresolvable circular reference?

Solution

There are many solution to this problem.

Redesign

Can you redesign your classes to avoid circular dependencies. Circular dependencies are bad and should be avoided if possible at all. There are very rare situations where circular dependencies can not be avoid.

Setter injection

If you use Setter injection instead of Constructor injection to solve Circular dependencies in Spring, there won’t be any issue.
Let’s change the above example and try Setter injection.
Update A.java

Update B.java

Update ApplicationContext.xml

Run CircularDependencyMain again
Output

Jun 15, 2018 11:35:26 AM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@31cefde0: startup date [Fri Jun 15 11:35:26 IST 2018]; root of context hierarchy
Jun 15, 2018 11:35:26 AM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [ApplicationContext.xml] Doing some work
Jun 15, 2018 11:35:26 AM org.springframework.context.support.ClassPathXmlApplicationContext doClose
INFO: Closing org.springframework.context.support.ClassPathXmlApplicationContext@31cefde0: startup date [Fri Jun 15 11:35:26 IST 2018]; root of context hierarchy

@Lazy with @Autowired annotation

You use @Lazy with @Autowired to solve circular dependencies in Spring.
Let’s change the above example and try with @Lazy annotation.
Update A.java
Please note that we have used @Autowired and @Lazy annotations here.

Update B.java

Update ApplicationContext.xml

Run CircularDependencyMain again
Output

Jun 15, 2018 11:35:26 AM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@31cefde0: startup date [Fri Jun 15 11:35:26 IST 2018]; root of context hierarchy
Jun 15, 2018 11:35:26 AM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [ApplicationContext.xml] Doing some work
Jun 15, 2018 11:35:26 AM org.springframework.context.support.ClassPathXmlApplicationContext doClose
INFO: Closing org.springframework.context.support.ClassPathXmlApplicationContext@31cefde0: startup date [Fri Jun 15 11:35:26 IST 2018]; root of context hierarchy

As you can see, program works fine with @Lazy annotation as well.

@PostConstruct with @Autowired annotation

You use @PostConstruct with @Autowired to solve circular dependencies in Spring.
Let’s change the above example and try Setter injection.
Update A.java
Please note that we have used @PostConstruct and @Autowired annotations here.

Update B.java

Update ApplicationContext.xml

Run CircularDependencyMain again
Output

Jun 15, 2018 11:35:26 AM org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@31cefde0: startup date [Fri Jun 15 11:35:26 IST 2018]; root of context hierarchy
Jun 15, 2018 11:35:26 AM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [ApplicationContext.xml] Doing some work
Jun 15, 2018 11:35:26 AM org.springframework.context.support.ClassPathXmlApplicationContext doClose
INFO: Closing org.springframework.context.support.ClassPathXmlApplicationContext@31cefde0: startup date [Fri Jun 15 11:35:26 IST 2018]; root of context hierarchy

As you can see, program works fine with @PostContruct annotation as well.

Thats’s all about Circular dependencies in Spring.
For more questions, refer Spring interview questions.

Was this post helpful?

Leave a Reply

Your email address will not be published. Required fields are marked *