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(格式化)的主要内容,如果未能解决你的问题,请参考以下文章
Spring官网阅读(十五)Spring中的格式化(Formatter)