Перейти к основному содержимому

Аннотация @Primary в Spring

· 3 мин. чтения

1. Обзор

В этом кратком руководстве мы обсудим аннотацию Spring @Primary , которая была представлена в версии 3.0 фреймворка.

Проще говоря, мы используем @Primary , чтобы отдавать предпочтение компоненту, когда имеется несколько компонентов одного типа.

Опишем проблему подробно.

2. Зачем нужен @Primary ?

В некоторых случаях нам нужно зарегистрировать более одного bean-компонента одного типа .

В этом примере у нас есть bean- компоненты JohnEmployee() и TonyEmployee() типа Employee :

@Configuration
public class Config {

@Bean
public Employee JohnEmployee() {
return new Employee("John");
}

@Bean
public Employee TonyEmployee() {
return new Employee("Tony");
}
}

Spring выбрасывает исключение NoUniqueBeanDefinitionException , если мы пытаемся запустить приложение .

Для доступа к bean-компонентам одного типа мы обычно используем аннотацию @Qualifier("beanName") .

Применяем его в точке инъекции вместе с @Autowired . В нашем случае мы выбираем bean-компоненты на этапе настройки, поэтому здесь нельзя применить @Qualifier . Подробнее об аннотации @Qualifier мы можем узнать , перейдя по ссылке .

Для решения этой проблемы Spring предлагает аннотацию @Primary .

3. Используйте @Primary с @Bean

Давайте посмотрим на класс конфигурации:

@Configuration
public class Config {

@Bean
public Employee JohnEmployee() {
return new Employee("John");
}

@Bean
@Primary
public Employee TonyEmployee() {
return new Employee("Tony");
}
}

Мы помечаем bean-компонент TonyEmployee() с помощью @Primary . Spring будет внедрять bean-компонент TonyEmployee() предпочтительно, а не JohnEmployee() .

Теперь давайте запустим контекст приложения и получим из него bean-компонент Employee :

AnnotationConfigApplicationContext context
= new AnnotationConfigApplicationContext(Config.class);

Employee employee = context.getBean(Employee.class);
System.out.println(employee);

После запуска приложения:

Employee{name='Tony'}

Из вывода мы видим, что экземпляр TonyEmployee() имеет предпочтение при автоподключении .

4. Используйте @Primary с @Component

Мы можем использовать @Primary непосредственно для bean-компонентов . Давайте рассмотрим следующий сценарий:

public interface Manager {
String getManagerName();
}

У нас есть интерфейс менеджера и два bean-компонента подкласса, DepartmentManager :

@Component
public class DepartmentManager implements Manager {
@Override
public String getManagerName() {
return "Department manager";
}
}

И бин GeneralManager :

@Component
@Primary
public class GeneralManager implements Manager {
@Override
public String getManagerName() {
return "General manager";
}
}

Оба они переопределяют getManagerName() интерфейса Manager . Также обратите внимание, что мы помечаем bean-компонент GeneralManager с помощью @Primary .

На этот раз @Primary имеет смысл только тогда, когда мы включаем сканирование компонентов :

@Configuration
@ComponentScan(basePackages="org.foreach.primary")
public class Config {
}

Давайте создадим сервис для использования внедрения зависимостей при поиске нужного bean-компонента:

@Service
public class ManagerService {

@Autowired
private Manager manager;

public Manager getManager() {
return manager;
}
}

Здесь оба bean -компонента, DepartmentManager и GeneralManager , имеют право на автоматическое связывание.

Поскольку мы пометили компонент GeneralManager с помощью @Primary , он будет выбран для внедрения зависимостей :

ManagerService service = context.getBean(ManagerService.class);
Manager manager = service.getManager();
System.out.println(manager.getManagerName());

Результатом является « Генеральный менеджер».

5. Вывод

В этой статье мы узнали об аннотации Spring @Primary . С помощью примеров кода мы продемонстрировали необходимость и варианты использования @Primary.

Как обычно, полный код для этой статьи доступен в проекте GitHub .