Spring Boot : @Primary and @Qualifier
Let's say we have a PaymentProcessorService in our application. This service needs to send notifications to users and for this, the class requires the NotificationManagementService type instance to be injected.
The NotificationManagementService has three implementations: SMSNotificationManagementService, PushNotificationManagementService, and EmailNotificationManagementService. Based on the use case, when we use PaymentProcessorService, we need to inject SMS, Push, and/or Email notification management services accordingly.
But will the following code work?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | @Service public class SMSNotificationManagementService implements NotificationManagementService { } @Service public class PushNotificationManagementService implements NotificationManagementService { } @Service public class EmailNotificationManagementService implements NotificationManagementService { } // INJECTING @Service public class PaymentProcessorService { private final NotificationManagementService notificationManagementService; @Autowired public PaymentProcessorService(NotificationManagementService notificationManagementService) { this.notificationManagementService = notificationManagementService; } } |
Answer is No
This will throw a NoUniqueBeanDefinitionException because Spring Boot finds multiple beans registered for NotificationManagementService and cannot decide which one to use.
So, it's obvious that we need a way to tell the application which bean to use and when. That's where the annotations @Primary and @Qualifier come in.
- @Primary: Use this on the bean that should be injected by default.
- @Qualifier("<bean_name>"): Use this to instruct the application to inject a specific bean other than the default. We need to provide the registered bean name as the @Qualifier parameter. (By default, Spring takes the class name, converts the first letter to lowercase and use it as bean name)
1 2 3 4 5 6 7 8 9 10 11 12 | @Service public class SMSNotificationManagementService implements NotificationManagementService { } @Service @Primary public class PushNotificationManagementService implements NotificationManagementService { } @Service public class EmailNotificationManagementService implements NotificationManagementService { } |
Injecting Beans with @Primary and @Qualifier
By default, PushNotificationManagementService will be injected because of @Primary
1 2 3 4 | @Autowired public PaymentProcessorService(NotificationManagementService notificationManagementService) { this.notificationManagementService = notificationManagementService; } |
To inject SMSNotificationManagementService, we use @Qualifier
1 2 3 4 5 6 | @Autowired public PaymentProcessorService( @Qualifier("sMSNotificationManagementService") NotificationManagementService notificationManagementService ) { this.notificationManagementService = notificationManagementService; } |
To inject EmailNotificationManagementService, we specify it with @Qualifier
1 2 3 4 5 6 | @Autowired public PaymentProcessorService( @Qualifier("emailNotificationManagementService") NotificationManagementService notificationManagementService ) { this.notificationManagementService = notificationManagementService; } |
By this way we reduce ambiguidy and define which bean to be injected explicitly. Also if we used custom bean names, we have to use those names within Qualifier annotation to define the bean to be injected.
Happy Coding 🙌