Spring ConversionService добавление конвертеров
Я искал следующую проблему, но не смог найти ответ.
Я хочу использовать сервис преобразования Spring, написав свой собственный конвертер, который реализует org.springframework.core.convert.converter.Converter.
Затем я добавляю свой собственный конвертер, как показано ниже:
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
<property name="converters">
<list>
<bean id="StringToLoggerConverter" class="com.citi.spring.converter.LoggerConverter" />
</list>
</property>
</bean>
При выполнении вышеизложенного моя инициализация приложения завершается неудачно, потому что я переопределяю службу преобразования bean и регистрирую только свой пользовательский конвертер.
Как я могу не переопределить convertService и только добавить свой собственный конвертер в список конвертеров, сохраняя при этом существующие?
Заранее спасибо.
10 ответов
Экспериментируя с разными способами и даже следуя некоторым весенним исходным кодам, я натолкнулся на интересную вещь.
Единственный способ, который я обнаружил, - это использование ConversionService без переопределения существующих конвертеров моими собственными, - это либо расширить, либо заново реализовать конверсионный сервис, вызвав метод afterPropertiesSet() суперкласса, чтобы зарегистрировать конвертеры по умолчанию, а затем добавить собственные.
Но даже если бы я использовал этот способ, во время выполнения я получал исключение, что не был найден конвертер для моих конкретных типов (например, из String в Logger).
Это вызвало у меня интерес, и я следовал исходному коду Spring, чтобы выяснить, почему, и я понял, что Spring пытается найти пользовательский конвертер, зарегистрированный в PropertyEditor. Я не уверен, почему это происходит. Я должен добавить здесь, что мое приложение не использует Spring MVC и ConversionService может как-то зарегистрироваться, и я этого не делал.
Наконец, я решил проблему с регистрацией пользовательского конвертера с помощью редактора свойств. Эта документация может рассматриваться как ссылка:
http://static.springsource.org/spring/docs/current/spring-framework-reference/html/validation.html
Мне было бы очень интересно узнать, почему Spring не находит мои зарегистрированные пользовательские конвертеры в реестре convertService (или, по крайней мере, почему Spring не просматривает этот реестр, чтобы найти пользовательские конвертеры). Я пропустил какую-либо конфигурацию?
Для тех, кто сталкивается с этим сейчас с помощью поиска в Google или аналогичных 2+ лет после того, как вопрос был первоначально опубликован, добавление конвертеров стало намного проще с помощью Java Config: WebMvcConfigurerAdapter
обеспечивает addFormatters(FormatterRegistry)
метод, который можно использовать для указания дополнительных пользовательских конвертеров.
Вы также можете добавить его динамически, используя метод addConverter в вашем классе DefaultConversionService-ish:
DefaultConversionService cs = new <YourClassThatInheritsFromDefaultConversionService or DefaultConversionService>();
cs.addConverter(new MyConverter());
Чтобы разрешить любые циклические зависимости, вот шаги, которые я выполнил (Spring Boot v5.2.1):
Зарегистрируйте простую конверсионную службу
@Configuration
public class ConverterProvider {
@Bean
public ConversionService conversionService() {
ConversionService conversionService = new GenericConversionService();
return conversionService;
}
}
Добавьте свои собственные конвертеры
@Component
public class ConvertersInjection {
@Autowired
private GenericConversionService conversionService;
@Autowired
private void converters(Set<Converter> converters) {
converters.forEach(conversionService::addConverter);
}
}
Конвертер может даже автоматически подключить вашу конверсионную службу
@Component
public class PushNotificationConverter implements Converter<PushNotificationMessages.PushNotification, GCM> {
@Lazy
@Autowired
private ConversionService conversionService;
@Override
public GCM convert(PushNotificationMessages.PushNotification source) {
GCM gcm = new GCM();
if (source.hasContent()) {
PushNotificationMessages.PushNotification.Content content = source.getContent();
if (content.hasData()) {
conversionService.convert(content.getData(), gcm.getData().getClass());
} else if (content.hasNotification()) {
conversionService.convert(content.getNotification(), gcm.getNotification().getClass());
}
}
return gcm;
}
}
С пружиной> 4 больше нет необходимости реализовывать собственный производный от ConversionService. Инициализируйте его в аннотированном классе @Configuration следующим образом:
@Configuration
public class ConversionServiceProvider
{
@Autowired
private MyConverterImpl myConverter;
@Bean
public ConversionService getConversionService()
{
ConversionServiceFactoryBean bean = new ConversionServiceFactoryBean();
bean.setConverters( getConverters() );
bean.afterPropertiesSet();
ConversionService object = bean.getObject();
return object;
}
private Set<Converter<?, ?>> getConverters()
{
Set<Converter<?, ?>> converters = new HashSet<Converter<?, ?>>();
converters.add( myConverter );
// add here more custom converters, either as spring bean references or directly instantiated
return converters;
}
}
Этот вариант у меня работает. Если вы используете конфигурацию java, вы можете добавить свои конвертеры в существующиеGenericConversionService
Config methods may have an arbitrary name and any number of arguments; each of those arguments will be autowired with a matching bean in the Spring container. Bean property setter methods are effectively just a special case of such a general config method. Such config methods do not have to be public.
@Configuration
class MyConfig {
@Autowired
void conversionService(GenericConversionService genericConversionService) {
genericConversionService.addConverter(String.class, UUID.class, UUID::fromString);
genericConversionService.addConverter(String.class, DateTime.class, DateTime::parse);
genericConversionService.addConverter(String.class, EnumState.class, EnumState::valueOf);
}
}
Наткнулся на очень интересный способ сделать это в Stackru - /questions/43111518/est-li-sposob-povtorno-ispolzovat-util-list-v-fajle-konfiguratsii-spring/43111528#43111528
Используя функцию под названием " Слияние коллекций", вы можете сделать это:
<bean id="customConversionService" class="org.springframework.context.support.ConversionServiceFactoryBean" parent="conversionService">
<property name="converters">
<list merge="true">
<bean id="StringToLoggerConverter" class="com.citi.spring.converter.LoggerConverter" />
</list>
</property>
</bean>
Достаточно переопределить ConversionServiceFactoryBean.afterPropertiesSet() и установить объект ConversionService для ваших конвертеров там. Пусть ваши конвертеры реализуют некоторый интерфейс, который позволяет устанавливать объект ConversionService, например ConversionServiceAware. Единственная проблема - получить доступ к зарегистрированным конвертерам, поэтому вам также придется переопределить метод setConverters().
public class MyFormattingConversionServiceFactoryBean extends FormattingConversionServiceFactoryBean {
private Set<?> cachedConverters = new LinkedHashSet<>();
@Override
public void setConverters(Set<?> converters) {
super.setConverters(converters);
this.cachedConverters = new LinkedHashSet<>(converters);
}
@Override
public void afterPropertiesSet() {
super.afterPropertiesSet();
FormattingConversionService conversionService = getObject();
for (Object converter : cachedConverters) {
if (converter instanceof ConversionServiceAware) {
((ConversionServiceAware) converter).setConversionService(conversionService);
}
}
}
}
Написать пользовательский сервис преобразования
public class CustomerConversionServiceFactoryBean extends ConversionServiceFactoryBean {
List<Converter<Object, Object>> converters = new ArrayList<>();
public CustomerConversionServiceFactoryBean() {
super();
DefaultConversionService conversionservice = (DefaultConversionService) super.getObject();
for(int i=0;i<converters.size();i++){
conversionservice.addConverter(converters.get(i));
}
}
}
затем измените свой XML
<bean id="conversionService"
class="CustomerConversionServiceFactoryBean" >
<property name="converters" >
<list>
<bean class="CustomConverter" />
</list>
</property>
</bean>
Я думаю, что это поможет вам...
В мой очень-очень старый код были добавлены преобразователи в этом классе. Чтобы решить мою проблему, мне пришлось добавить еще один преобразователь, просто добавив дополнительные строки в тот же класс.addFormatters
(переопределенный) метод.WebMvcContext implements WebMvcConfigurer
Линии:
@Override
public void addFormatters(FormatterRegistry registry) {
.....
registry.addConverter( SourceType.class, TargetType.class, converter);