关于日期转化的 文章很多.我只要讲讲我遇到的一些与日期转化有关的问题.
大约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;
- }
- }
还要进行一些配置,具体参见:
date java.lang.IllegalArgumentException: argument type mismatch
这里我要强调一点:我之所以写converter是为了根本上解决问题,这样不管浏览器是什么语言都不用担心了.
我推荐使用org.apache.commons.lang3.time.DateUtils 来进行日期转化,为什么呢?
我们看到org.apache.commons.lang3.time.DateUtils 的第二个参数是一个数组,意思就是我循环的从数组中拿出一个pattern进行parse,直到解析成功为止.其实这就是一种容错的思维,非常好的思维.
后来在spring MVC中也遇到了关于日期转化的问题
场景:更新的时候,需要把日期放到表单的隐藏域中,但是如果日期为空的话,提交就报错,大概意思:
can not convert String "" to Date.根本原因是隐藏域的值为空.
所以解决方法:也是写一个转化器,如果日期字符串为空,则返回空就完了,不必报错.