Перевод подготовлен специально для будущих студентов курса "Разработчик на Spring Framework".
Эта статья о динамическом создании бинов за пять лет стала самой популярной в моем блоге (более 9300 просмотров). Пришло время ее обновить. Также я добавил пример на Github.
Однажды на тренинге меня спросили: "Можно ли создать Spring Bean динамически, чтобы можно было выбрать реализацию во время выполнения". Так как во время компиляции еще не известно, какой бин должен быть создан. Приложение должно решить это на основе properties-файла.
1. Создадим аннотацию для того, чтобы отметить метод, который должен создавать объект динамически:
package your.package;
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectDynamicObject {
}
2. Далее используем ее в методе, который должен создать объект:
@Service
public class CustomerServiceImpl {
private Customer dynamicCustomerWithAspect;
@InjectDynamicObject
public Customer getDynamicCustomerWithAspect() {
return this.dynamicCustomerWithAspect;
}
}
3. Напишем аспект с Pointcut и Advise, который изменяет объект, возвращаемый методом на шаге 2:
@Component
@Aspect
public class DynamicObjectAspect {
// This comes from the property file
@Value("${dynamic.object.name}")
private String object;
@Autowired
private ApplicationContext applicationContext;
@Pointcut("execution(@com.lofi.springbean.dynamic.
InjectDynamicObject * *(..))")
public void beanAnnotatedWithInjectDynamicObject() {
}
@Around("beanAnnotatedWithInjectDynamicObject()")
public Object adviceBeanAnnotatedWithInjectDynamicObject(
ProceedingJoinPoint pjp) throws Throwable {
pjp.proceed();
// Create the bean or object depends on the property file
Object createdObject = applicationContext.getBean(object);
return createdObject;
}
}
4. Пишем класс, который должен возвращаться из @InjectDynamicObject. Имя класса настраивается в properties-файле. В данном примере я написал две реализации Customer: CustomerOneImpl
и CustomerTwoImpl:
@Component("customerOne")
public class CustomerOneImpl implements Customer {
@Override
public String getName() {
return "Customer One";
}
}
application.properties
dynamic.object.name=customerOne
5. Пишем тест:
@RunWith(SpringRunner.class)
@SpringBootTest
public class CustomerServiceImplTest {
@Autowired
private CustomerServiceImpl customerService;
@Test
public void testGetDynamicCustomerWithAspect() {
// Dynamic object creation
logger.info("Dynamic Customer with Aspect: " +
customerService.getDynamicCustomerWithAspect()
.getName());
}
Но есть еще, более простой, способ сделать это. Без аспектов и AspectJ, только чистый Spring. Можно просто сохранить все реализации в Map и получить из нее необходимую реализацию. Так мы сделали в приложении eXTra Client. В качестве примера можно посмотреть на реализацию PluginsLocatorManager. Spring автоматически инжектит Map с именем бина (String) и самим бином.
"… Даже типизированные Map можно инжектить автоматически, если тип ключа String. В значениях Map будут все бины ожидаемого типа, а в ключах — соответствующие имена бинов".
Подробнее см. в документации Spring.
@Service
public class CustomerServiceImpl {
// We inject the customer implementations into a Map
@Autowired
private Map<String, Customer> dynamicCustomerWithMap;
// This comes from the property file as a key for the Map
@Value("${dynamic.object.name}")
private String object;
public Customer getDynamicCustomerWithMap() {
return this.dynamicCustomerWithMap.get(object);
}
}
Подробнее о курсе "Разработчик на Spring Framework" можно узнать здесь.
ultrinfaern
Если вам нужен бин в зависимости от проперти то лучше использовать @ConditionalOnProperty аннотацию.
Пример выше можно сделать, добавляя аннотацию @ConditionalOnProperty над каждым классом, который может там использоваться и значение проперти в аннотации это само название этого класса.