Converter(转换器)与Formatter(格式化)

Posted OverZeal

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Converter(转换器)与Formatter(格式化)相关的知识,希望对你有一定的参考价值。

Converter(转换器)与Formatter(格式化)都可以用于将一种对象类型转换为另一种对象类型。Converter是通用元件,可以在应用程序的任意层中使用,而Fotermatter这是专门为Web层设计的。Validator(验证器)主要用于校验输入。

Converter(转换器)

创建Converter,必须编写实现org.springframework.core.convert.converter.Converter接口的一个Java类。该接口的实现声明如下:

public interface Converter<S,T>

这里的S标识源类型,T表示目标类型。这是一个将String转为Employee对象的转换器:

public class EmployeeConveter implements Converter<String,Employee> {}

为了使用springMVC应用程序中定制的Converter,需要在springMVC配置文件中编写一个名为conversionService的bean。bean的类名称必须为org.springframeword.context.support.ConversionServiceFactoryBean。这个bean必须包含一个converters属性,它将列出要在应用程序中使用的所有定制Converter。

<!-- 配置自定义类型转换器 -->
<bean id="employeeConveter" class="cn.lynu.converter.EmployeeConveter"></bean>
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">
   <property name="converters">
     <set>
       <ref bean="employeeConveter"/>
     </set>
   </property>
</bean>

注意:不要忘了将这个employeeConveter 加入到IOC的控制中:<bean id="employeeConveter" class="cn.lynu.converter.EmployeeConveter"></bean>  亦或者使用@Component 注解配置该类加入到IOC容器中,这样就不需要在XML中配置了:

 

二者必须选择一种配置方法,将employeeConveter 加入到IOC容器中,因为我们在设置converters 的时候需要用到,不然就会报出找不到这个bean的异常:

随后,要是annotation-driver元素的conversion-service属性赋值bean名称,如下:

<mvc:annotation-driven conversion-service="conversionService"></mvc:annotation-driven>

Formatter(格式化)

Formatter就像Converter一样,也是将一种类型转换为另一种类型。但是,Formatter的源类型必须是一个String,而Converter则适用用于任意源类型。Formatter更适合Web层,而Converter则可以用在任意层,为了转换SpringMVC应用程序表单中的用户输入,始终应该选择Formatter,而不是Converter。

为了创建Formatter,要编写一个实现org.springframework.format.Formatter接口的Java类。下面是这个接口的声明:

public interface Formatter<T>

这里的T表示输入字符串要转换的目标类型。

public class LocalDateFormatter implements Formatter<LocalDate> {}

该接口有parse和print两个方法。所有实现都必须覆盖它们。

T parse(String text,java.util.Locale locale)

String print(T object,java.util.Local local)

parse方法利用值当的Locale将一个String解析为String解析成目标类型。print方法与之相反,他返回目标对象的字符串表示法。

为了在springMVC中使用Formatter,需要利用名为conversionService的bean对它进行注册。bean的类名称必须为org.springframework.format.support.FormatterConversionServiceFactoryBean,这个bean可以用一个·formatters属性注册formatter,用一个converters属性注册converter。

    <bean id="conversionService"
        class="org.springframework.format.support.FormattingConversionServiceFactoryBean">
        <property name="formatters">
            <set>
                <!--注册formatters-->
            </set>  
        </property>
    </bean>  

注意:formatters/converters都是set类型,我们还需要给这个Formatter添加一个component-scan元素:

<context:component-scan base-package="cn.lynu.formatter" />

或者是使用@Component注解

随后,还要是annotation-driver元素的conversion-service属性赋值bean名称,如下:

<mvc:annotation-driven conversion-service="conversionService" />

Validator (验证器)

在springMVC中,有两种方式可以验证输入,即利用Spring自带的验证框架,或者利用JSR303实现。可以这样理解:Converter和Formatter作用于字段级。在MVC应用程序中,它们将一种类型转换为另一种;类型。而验证器则作用对象级,它决定某一对象的所有字段是否均有效,以及是否遵循某些规则。

一个典型的springMVC应用会同时用到formatters/converters和validators。

如果一个应用程序即使用了Formatter,又有validator(验证器),那么,应用中的事件顺序是这样的:在调用Controller期间,将会有一个或者多个Formatter,试图将输入字符串转换为domain对象中的field值。一但格式化成功,验证器就会开始介入。

Spring验证器

从一开始,Spring就设计了输入验证,甚至早于JSR303(Java验证规范),因此,因此可能一些老的springMVC项目还在使用,对于新的项目,还是建议使用JSR303

为了创建Spring验证器,要实现org.springframework.validation.Validator接口,并重写supports和validate两个方法。

package org.springframework.validation;
public interface Validator{
        boolean supports(Class<?> clazz);
        void validate(Object target,Errors error);
}

如果验证器可以指定的Class,supports方法将返回true。validate方法会验证目标对象,并将错误消息填入Errors对象。

Errors对象是org.springframework.validation.Errors接口的一个实例。Errors对象中包含一系列FieldError和ObjectError对象。FieldError表示被验证对象中的某个属性相关的错误。

编写验证器时,不需要直接创建Error对象,因为实例化ObjectError或FieldError花费了大量编程精力。这是因为ObjectError类的构造器需要4个参数,FieldError类构造器则需要7个参数。

Errors对象中的错误消息,可以利用springMVC表单标签库的Errors标签显示在html页面中。错误消息可以通过Spring支持国际化特征本地化(springMVC表单标签,国际化本地化随后再说)。

org.springframework.validation.ValidationUtis类是一个工具,有助于编写Spring验证器,我们直接使用它里面提供的验证方法完成Validator类的验证过程。

配置文件

验证器不需要显示注册,但是想要从某个属性文件中获得错误消息,则需要通过声明messagesSource bean,告诉spring要去哪里寻找这个文件:

<bean id="messageSource" class="org.springframework.context.support.ReloadableResourceBundleMessageSource">
  <property name="basenames">
    <list>
      <value>classpath:messages</value>
    </list>
  </property>
  <property name="defaultEncoding" value="UTF-8"/>
  <property name="cacheSeconds" value="120"/>  
</bean>

建议都配上defaultEncoding,并将其设置为UTF-8,这样读取properties文件内容到页面显示就不会有乱码出现了。而且如果使用Maven构建项目时,properties文件如果放在src/main/resources下,都建议在写着文件路径的时候都加上classpath。

所有的验证出错信息都将放在BindingResult对象中,我们可以调用其方法判断验证时候出错:

if(bindingResult.hasErrors()){
     //有错误出现
            System.out.println("错误个数:"+bindingResult.getErrorCount());
            System.out.println("错误为:");
            for(FieldError error:bindingResult.getFieldErrors()) {
                System.out.println(error.getField()+"\\t"+error.getDefaultMessage());
            }
}

其实这个BindingResult对象使用是有要求的,我们在下面介绍过JSR303之后再说

JSR303验证

JSR303通过租借给对象属性添加约束,但是JSR303只是一个规范,所以我们需要使用它的实现产品,目前两个实现:第一个时Hibernate Validator,第二个实现时Apache BVal。JSR303方便之处就在于不需要编写验证器,但要利用JSR303注解类型潜入约束。这些注解大都是javax.validation包下的类.

部分JSR303验证注解:

注解 描述 范例
@AssertTrue 应用于boolean属性,该属性必须为True

@AssertTrue

boolean isEmpty;

@DecimalMax 该属性的值必须为小于或等于指定值的小数

@DecimalMax("1.1")

BigDecimal price;

@Max 该属性值必须为一个小于或等于指定值的整数

@Max(150)

int age;

@Future 该属性必须为未来的一个日期  

@Future

Date shippingDate;

@Past 该属性值必须是过去的一个日期

@Past

Date birthDay;

@Size 该属性值必须在指定范围类

@Size(min=2,max=140)

String areaCode;

@NotNull

该属性值不能是null(建议使用Hibernate Validation的@NotEmpty,

因为页面字段为空会被封装为[] 而非null)

@NotNull

String firstName;

@Patter

该属性可以与指定的表达式相匹配(其实就是与正则匹配)

@Pattern(regext="\\\\d{3}")

String areaCode;

我们还是需要在springMVC中配置ReloadableResourceBundleMessageSource:

<bean id="messageSource" class="org.springframework.context.support.ReloadableResourceBundleMessageSource">
  <property name="basenames">
    <list>
      <value>classpath:messages</value>
    </list>
  </property>
  <property name="defaultEncoding" value="UTF-8"/>
  <property name="cacheSeconds" value="120"/>  
</bean>

像使用Spring验证器一样,可以在messages.properties属性文件以下面格式来使用property键值,覆盖来自JSR303验证器的错误消息:

constraint.object.property

但是在5.2.4版本Hibernate Validation仍然无法验证使用@Part和@Future注解的LocalDate或LocalDateTime类型的字段,因为LocalDate/LocalDateTime是Java8新增的类型,是想用来替代java.util.Date。因此对于这种新的类型的验证需要使用第一种自定义spring验证器方式,或者是重写Hibernate Validation中的ConstraintHelper类。

JSR303验证的后台Java代码

注意:我们上面说到所有验证错误信息都会存在BindingResult这个对象中,但是这个对象使用是有要求的:它必须在放在验证对象(验证对象需要使用@Valid注解修饰 javax.validation.Valid)的后面,必须在后面,也就是说它们之间不能存在其他任何的参数:

    /**
     * BindingResult将显示错误信息
     */
    @RequestMapping(value="/emp",method=RequestMethod.POST)
    public String save(@Valid Employee employee,BindingResult result,Model model) {
        if(result.getErrorCount()>0) {
            System.out.println("错误个数:"+result.getErrorCount());
            System.out.println("错误为:");
            for(FieldError error:result.getFieldErrors()) {
                System.out.println(error.getField()+"\\t"+error.getDefaultMessage());
            }
               //输出错误信息
               List<ObjectError> allErrors = result.getAllErrors();
               List<String> errorList=new ArrayList<String>();
                for (ObjectError objectError : allErrors) {
                    try {
                        errorList.add(objectError.getDefaultMessage());
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
                //将错误信息传递到页面
            model.addAttribute("allErrors", errorList);
            model.addAttribute("departments", departmentDao.getDepartments());
            return "input";
        }
        employeeDao.save(employee);
        return "redirect:/list";
    }

 

以上是关于Converter(转换器)与Formatter(格式化)的主要内容,如果未能解决你的问题,请参考以下文章

SpringMVC的格式化转换器Formatter

spring类型转换器(三)

Spring官网阅读(十五)Spring中的格式化(Formatter)

Spring读源码系列番外篇04----类型转换--上

进行物理机与虚拟机转换时,运行VMware Converter时出现错误,求高手解决

System.Text.Json 自定义Converter实现时间转换