136

我正在寻找在 Spring MVC 中绑定和转换数据的最简单和最简单的方法。如果可能,不做任何 xml 配置。

到目前为止,我一直在使用PropertyEditors

public class CategoryEditor extends PropertyEditorSupport {

    // Converts a String to a Category (when submitting form)
    @Override
    public void setAsText(String text) {
        Category c = new Category(text);
        this.setValue(c);
    }

    // Converts a Category to a String (when displaying form)
    @Override
    public String getAsText() {
        Category c = (Category) this.getValue();
        return c.getName();
    }

}

...
public class MyController {

    @InitBinder
    public void initBinder(WebDataBinder binder) {
        binder.registerCustomEditor(Category.class, new CategoryEditor());
    }

    ...

}

很简单:两种转换都定义在同一个类中,并且绑定很简单。如果我想对所有控制器进行通用绑定,我仍然可以在我的 xml 配置中添加 3 行


但是 Spring 3.x 引入了一种新方法,使用Converters

在 Spring 容器中,该系统可以用作 PropertyEditors 的替代品

因此,假设我想使用 Converters,因为它是“最新的替代方案”。我将不得不创建两个转换器:

public class StringToCategory implements Converter<String, Category> {

    @Override
    public Category convert(String source) {
        Category c = new Category(source);
        return c;
    }

}

public class CategoryToString implements Converter<Category, String> {

    @Override
    public String convert(Category source) {
        return source.getName();
    }

}

第一个缺点:我必须上两节课。好处:由于通用性,无需转换。

那么,我如何简单地对转换器进行数据绑定?

第二个缺点:我还没有找到任何简单的方法(注释或其他编程工具)在控制器中执行此操作:与someSpringObject.registerCustomConverter(...);.

我发现的唯一方法是乏味的,并不简单,而且只涉及一般的跨控制器绑定:

  • XML配置

    <bean id="conversionService"
      class="org.springframework.context.support.ConversionServiceFactoryBean">
        <property name="converters">
            <set>
                <bean class="somepackage.StringToCategory"/>
                <bean class="somepackage.CategoryToString"/>
            </set>
        </property>
    </bean>
    
  • Java 配置仅在 Spring 3.1+ 中):

    @EnableWebMvc
    @Configuration
    public class WebConfig extends WebMvcConfigurerAdapter {
    
        @Override
        protected void addFormatters(FormatterRegistry registry) {
            registry.addConverter(new StringToCategory());
            registry.addConverter(new CategoryToString());
        }
    
    }
    

有了所有这些缺点,为什么要使用 Converters ?我错过了什么吗?还有其他我不知道的技巧吗?

我很想继续使用 PropertyEditors... 绑定更容易和更快。

4

4 回答 4

59

有了所有这些缺点,为什么要使用 Converters ?我错过了什么吗?还有其他我不知道的技巧吗?

不,我认为您已经非常全面地描述了 PropertyEditor 和 Converter,以及如何声明和注册每一个。

在我看来,PropertyEditor 的范围是有限的——它们有助于将 String 转换为一种类型,而这个字符串通常来自 UI,因此使用 @InitBinder 和 WebDataBinder 注册 PropertyEditor 是有意义的。

另一方面,转换器更通用,它适用于系统中的任何转换 - 不仅仅是 UI 相关的转换(字符串到目标类型)。例如,Spring Integration 广泛使用转换器将消息负载转换为所需的类型。

我认为对于 UI 相关的流程,PropertyEditors 仍然适用,特别是在您需要为特定命令属性进行自定义操作的情况下。对于其他情况,我会从 Spring 参考中获取建议并改为编写转换器(例如,将 Long id 转换为实体,例如)。

于 2012-09-22T20:26:45.303 回答
17
  1. 对于 to/from 字符串转换使用格式化程序(实现org.springframework.format.Formatter)而不是转换器。它有print(...)parse(...)方法,所以你只需要一个类而不是两个。要注册它们,请使用FormattingConversionServiceFactoryBean,它可以注册转换器和格式化程序,而不是ConversionServiceFactoryBean
  2. 新的 Formatter 东西有几个额外的好处:
    • Formatter 接口在其print(...)parse(...)方法中提供 Locale 对象,因此您的字符串转换可以是区域敏感的
    • 除了预先注册的格式化程序之外,FormattingConversionServiceFactoryBean还附带了几个方便的预先注册的AnnotationFormatterFactory对象,允许您通过注释指定其他格式化参数。例如: @RequestParam@DateTimeFormat(pattern="MM-dd-yy")LocalDate baseDate ... 创建自己的AnnotationFormatterFactory类并不难,简单的例子见 Spring 的NumberFormatAnnotationFormatterFactory。我认为这消除了对特定于控制器的格式化程序/编辑器的需求。为所有控制器使用一个ConversionService并通过注释自定义格式。
  3. 我同意如果您仍然需要一些特定于控制器的字符串转换,最简单的方法仍然是使用自定义属性编辑器。(我试图在我的@InitBinder方法中调用' binder.setConversionService(...) ' ,但它失败了,因为binder 对象已经设置了'全局'转换服务。似乎不鼓励每个控制器的转换类弹簧 3)。
于 2014-04-10T08:16:54.673 回答
8

最简单(假设您使用的是持久性框架)但不是完美的方法是通过ConditionalGenericConverter接口实现通用实体转换器,该接口将使用其元数据转换实体。

例如,如果您使用 JPA,此转换器可能会查看指定的类是否具有@Entity注释,并使用@Id带注释的字段来提取信息并使用提供的字符串值作为查找的 Id 自动执行查找。

public interface ConditionalGenericConverter extends GenericConverter {
    boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType);
}

ConditionalGenericConverter是 Spring 转换 API 的“终极武器”,但一旦实现就可以处理大部分实体转换,从而节省开发人员的时间——当您只是将实体类指定为控制器的参数而从不考虑实现时,这是一种极大的解脱一个新的转换器(当然除了自定义和非实体类型)。

于 2012-09-22T15:14:53.330 回答
1

您可以通过将两个转换器实现为静态内部类来解决拥有两个单独的转换器类的需要。

public class FooConverter {
    public static class BarToBaz implements Converter<Bar, Baz> {
        @Override public Baz convert(Bar bar) { ... }
    }
    public static class BazToBar implements Converter<Baz, Bar> {
        @Override public Bar convert(Baz baz) { ... }
    }
}

您仍然需要分别注册它们,但如果您进行任何更改,至少这会减少您需要修改的文件数量。

于 2017-04-25T03:16:23.347 回答