关于日期转化的 文章很多.我只要讲讲我遇到的一些与日期转化有关的问题.
大约1年前,我遇到过一个很令人郁闷又费解的问题.
项目使用的是struts2框架,表单中有日期,提交表单时总是报错,说日期转化有问题,没能马上找到原因.
过了很长时间,才搞清楚关键之所在,有一个日期格式是:"yyyy-MM-ddTHH:mm:ss".
日期和时间之间使用T 分割,而不是采用我们熟知的空格来分割."yyyy-MM-ddTHH:mm:ss"是美国一种的日期格式,中文中是没有这种日期格式的.
我当时使用的浏览器是IE,碰巧当时IE浏览器的语言被设置成了英语,所以提交表单时struts2按照yyyy-MM-ddTHH:mm:ss来解析,但是我表单中日期的格式却是""yyyy-MM-dd?HH:mm:ss"",所以就报错了.
怎么办呢?不是仅仅把浏览器的语言设置成为中文就OK 了,因为这没有彻底解决问题.
我的方法是编写一个自定义的日期转化Converter,当然得继承struts2的ognl.DefaultTypeConverter
package com.common.convert; import java.sql.Date; import java.sql.Timestamp; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Map; import ognl.DefaultTypeConverter; import org.apache.commons.lang.StringUtils; import org.apache.commons.lang3.time.DateUtils; import org.apache.log4j.Logger; /*** * 日期转换器.<br>用于struts2 * * @author huangwei * */ public class DateConverter extends DefaultTypeConverter { private static final Logger logger = Logger.getLogger(DateConverter.class); private static final String DATETIME_PATTERN = "yyyy-MM-dd HH:mm:ss"; private static final String DATETIME_PATTERN_NO_SECOND = "yyyy-MM-dd HH:mm"; private static final String DATETIME_PATTERN_ZH ="yyyy年MM月dd日 HH点mm分ss秒"; private static final String DATETIME_PATTERN_ZH2 ="yyyy年MM月dd日 HH时mm分ss秒"; private static final String DATE_PATTERN = "yyyy-MM-dd"; private static final String MONTH_PATTERN = "yyyy-MM"; /** * Convert value between types */ @SuppressWarnings("unchecked") public Object convertValue(Map ognlContext, Object value, Class toType) { Object result = null; // java.sql.Date 是java.util.Date 的子类 if (toType == java.util.Date.class) { try { result = doConvertToDate(value, toType); } catch (ParseException e) { e.printStackTrace(); } } else if (toType == java.sql.Timestamp.class) { try { java.util.Date date=doConvertToDate(value, toType); result = new Timestamp(date.getTime()); } catch (ParseException e) { e.printStackTrace(); } } else if (toType == String.class) { result = doConvertToString(value); } return result; } /** * Convert String to Date * * @param value * @return * @throws ParseException */ private java.util.Date doConvertToDate(Object value,Class toType) throws ParseException { java.util.Date result = null; if (value instanceof String) { result = DateUtils.parseDate((String) value, new String[] {DATETIME_PATTERN, DATE_PATTERN, MONTH_PATTERN ,DATETIME_PATTERN_NO_SECOND,DATETIME_PATTERN_ZH,DATETIME_PATTERN_ZH2}); // if(toType==java.sql.Timestamp.class){ // result=new java.sql.Timestamp(result.getTime()); // } // all patterns failed, try a milliseconds constructor if (result == null && StringUtils.isNotEmpty((String) value)) { try { result = new Date(new Long((String) value).longValue()); } catch (Exception e) { logger.error("Converting from milliseconds to Date fails!"); e.printStackTrace(); } } } else if (value instanceof Object[]) { // let‘s try to convert the first element only Object[] array = (Object[]) value; if ((array != null) && (array.length >= 1)) { value = array[0]; result = doConvertToDate(value,toType); } } else if (Date.class.isAssignableFrom(value.getClass())) { result = (Date) value; } return (java.util.Date)result; } /** * Convert Date to String * * @param value * @return */ private String doConvertToString(Object value) { SimpleDateFormat simpleDateFormat = new SimpleDateFormat( DATETIME_PATTERN); String result = (String)value; if (value instanceof Date) { result = simpleDateFormat.format(value); } return result; } }
?还要进行一些配置,具体参见:?
这里我要强调一点:我之所以写converter是为了根本上解决问题,这样不管浏览器是什么语言都不用担心了.
我推荐使用org.apache.commons.lang3.time.DateUtils 来进行日期转化,为什么呢?
我们看到org.apache.commons.lang3.time.DateUtils 的第二个参数是一个数组,意思就是我循环的从数组中拿出一个pattern进行parse,直到解析成功为止.其实这就是一种容错的思维,非常好的思维.
?
后来在spring MVC中也遇到了关于日期转化的问题
场景:更新的时候,需要把日期放到表单的隐藏域中,但是如果日期为空的话,提交就报错,大概意思:
can not convert String "" to Date.根本原因是隐藏域的值为空.
所以解决方法:也是写一个转化器,如果日期字符串为空,则返回空就完了,不必报错.
?
原文:http://hw1287789687.iteye.com/blog/2156536