«1. Введение

В этом коротком уроке мы покажем, как динамически автомонтировать bean-компонент в Spring.

Мы начнем с того, что представим реальный пример использования, в котором может быть полезно динамическое автоматическое связывание. В дополнение к этому мы покажем, как решить ее в Spring двумя разными способами.

2. Варианты использования динамического автоматического связывания

Динамическое автоматическое связывание полезно в тех случаях, когда нам нужно динамически изменять логику выполнения компонента Spring. Это особенно удобно в тех случаях, когда код для выполнения выбирается на основе некоторых переменных времени выполнения.

Чтобы продемонстрировать реальный пример использования, давайте создадим приложение, которое управляет серверами в разных регионах мира. По этой причине мы создали интерфейс с двумя простыми методами:

public interface RegionService {
    boolean isServerActive(int serverId);

    String getISOCountryCode();
}

и двумя реализациями:

@Service("GBregionService")
public class GBRegionService implements RegionService {
    @Override
    public boolean isServerActive(int serverId) {
        return false;
    }

    @Override
    public String getISOCountryCode() {
        return "GB";
    }
}
@Service("USregionService")
public class USRegionService implements RegionService {
    @Override
    public boolean isServerActive(int serverId) {
        return true;
    }

    @Override
    public String getISOCountryCode() {
        return "US";
    }
}

Допустим, у нас есть веб-сайт, на котором пользователь может проверить, сервер активен в выбранном регионе. Следовательно, мы хотели бы иметь класс службы, который динамически изменяет реализацию интерфейса RegionService с учетом ввода пользователя. Несомненно, это тот случай, когда в игру вступает динамическое автоматическое связывание компонентов.

3. Использование BeanFactory

@Service
public class BeanFactoryDynamicAutowireService {
    private static final String SERVICE_NAME_SUFFIX = "regionService";
    private final BeanFactory beanFactory;

    @Autowired
    public BeanFactoryDynamicAutowireService(BeanFactory beanFactory) {
        this.beanFactory = beanFactory;
    }

    public boolean isServerActive(String isoCountryCode, int serverId) {
        RegionService service = beanFactory.getBean(getRegionServiceBeanName(isoCountryCode), 
          RegionService.class);

        return service.isServerActive(serverId);
    }

    private String getRegionServiceBeanName(String isoCountryCode) {
        return isoCountryCode + SERVICE_NAME_SUFFIX;
    }
}

BeanFactory — это корневой интерфейс для доступа к контейнеру компонентов Spring. В частности, он содержит полезные методы для получения конкретных бобов. Так как BeanFactory также является bean-компонентом Spring, мы можем автоматически подключать и использовать его непосредственно в нашем классе:

Мы использовали перегруженную версию метода getBean() для получения bean-компонента с заданным именем и желаемым типом.

И хотя это работает, мы действительно предпочли бы полагаться на что-то более идиоматичное; то есть что-то, что использует внедрение зависимостей.

4. Использование интерфейсов

Чтобы решить эту проблему с помощью внедрения зависимостей, мы будем полагаться на одну из менее известных функций Spring.

@Service
public class CustomMapFromListDynamicAutowireService {
    private final Map<String, RegionService> servicesByCountryCode;

    @Autowired
    public CustomMapFromListDynamicAutowireService(List<RegionService> regionServices) {
        servicesByCountryCode = regionServices.stream()
                .collect(Collectors.toMap(RegionService::getISOCountryCode, Function.identity()));
    }

    public boolean isServerActive(String isoCountryCode, int serverId) {
        RegionService service = servicesByCountryCode.get(isoCountryCode);

        return service.isServerActive(serverId);
    }
}

Помимо стандартной автосвязывания с одним полем, Spring дает нам возможность собирать все bean-компоненты, которые являются реализациями определенного интерфейса, в карту:

Мы создали карту в конструкторе, которая содержит реализации по их код страны. Кроме того, мы можем использовать его позже в методе, чтобы получить конкретную реализацию для проверки активности данного сервера в определенном регионе.

5. Заключение

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