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

https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/beans/factory/annotation/Autowired.html

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);
Другие вопросы по тегам