SpringMVC配置全局日期转换器,处理日期转换异常

EmiConybear 9年前

来自: http://blog.csdn.net//chenleixing/article/details/45156617


spring3.0配置日期转换可以通过配置自定义实现WebBingingInitializer接口的一个日期转换类来实现,方法如下

转换类:

  1. public class DateConverter implements WebBindingInitializer {    
  2.     
  3.     public void initBinder(WebDataBinder binder, WebRequest request) {    
  4.           
  5.         SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd");    
  6.         binder.registerCustomEditor(Date.classnew CustomDateEditor(df, false));    
  7.     }    

在spring-servlet.xml当中的进行注册:

  1. <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">    
  2.     <!-- 日期格式转换 -->    
  3.     <property name="webBindingInitializer">    
  4.         <bean class="DateConverter" />    
  5.     </property>    
  6. </bean>  

 

spring3.1.1的处理进行调整,所以按照3.0的写法在3.1.1里面是无效的,通过查找资料及测试,发现可行方法

原因:

annotation-driven缺省注册类的改变 

Spring 3.0.x中使用了annotation-driven后,缺省使用DefaultAnnotationHandlerMapping 来注册handler method和request的mapping关系。 AnnotationMethodHandlerAdapter来在实际调用handlermethod前对其参数进行处理。 

 

在spring mvc 3.1中,对应变更为 
DefaultAnnotationHandlerMapping -> RequestMappingHandlerMapping 
AnnotationMethodHandlerAdapter -> RequestMappingHandlerAdapter 
AnnotationMethodHandlerExceptionResolver -> ExceptionHandlerExceptionResolver 

以上都在使用了annotation-driven后自动注册。 
  而且对应分别提供了AbstractHandlerMethodMapping , AbstractHandlerMethodAdapter和 AbstractHandlerMethodExceptionResolver以便于让用户更方便的实现自定义的实现类。 

 

<mvc:annotation-driven/>相当于注册了DefaultAnnotationHandlerMapping和AnnotationMethodHandlerAdapter两个bean,配置一些messageconverter。即解决了@Controller注解的使用前提配置。

 

spring mvc <mvc:annotation-driven />会自动启动Spring MVC的注解功能,但实际它做了哪些工作呢?

 

Java代码   
  1. <bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping">  
  2. <property name="order" value="1" />  
  3. </bean>  
  4. <bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">  
  5. <property name="webBindingInitializer">  
  6.   <bean class="org.springframework.web.bind.support.ConfigurableWebBindingInitializer">  
  7.    <property name="conversionService" ref="conversionService" />  
  8.    <property name="validator" ref="validator" />  
  9.   </bean>  
  10. </property>  
  11. </bean>  
  12. <bean id="conversionService" class="org.springframework.samples.petclinic.util.PetclinicConversionServiceFactory" />  
  13. <bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean" />   


从上面的配置可以看出,我的配置应该是被sping的配置覆盖了,<mvc:annotation-driven />配置中已经包含了webBindingInitializer的配置,看来使用<mvc:annotation-driven />后与原来的配置出现了重复,这种情况下不管<mvc:annotation-driven />放在上面还是放在下面都会出现问题。

 

解决方法:

使用conversion-service来注册自定义的converter 
DataBinder实现了PropertyEditorRegistry, TypeConverter这两个interface,而在spring mvc实际处理时,返回值都是return binder.convertIfNecessary(见HandlerMethodInvoker中的具体处理逻辑)。因此可以使用customer conversionService来实现自定义的类型转换。 

<mvc:annotation-driven />中配置可以看出,AnnotationMethodHandlerAdapter已经配置了webBindingInitializer,我们可以通过设置其属性conversionService来实现自定义类型转换。

 

Java代码   
  1. <bean id="conversionService" class="org.springframework.format.support.FormattingConversionServiceFactoryBean">    
  2.         <property name="converters">    
  3.             <list>    
  4.                 <bean class="com.doje.XXX.web.DateConverter" />    
  5.             </list>    
  6.         </property>    
  7.     </bean>  

 

 需要修改spring service context xml配置文件中的annotation-driven,增加属性conversion-service指向新增的conversionService bean。 

 

Java代码   
  1. <mvc:annotation-driven conversion-service="conversionService" />   

 

 实际自定义的converter如下。 

Java代码   
  1. public class DateConverter implements Converter<String, Date> {    
  2. @Override    
  3. public Date convert(String source) {    
  4.     SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");    
  5.     dateFormat.setLenient(false);    
  6.     try {    
  7.         return dateFormat.parse(source);    
  8.     } catch (ParseException e) {    
  9.         e.printStackTrace();    
  10.     }           
  11.     return null;    
  12. }