java 日期转化

版权所有,禁止匿名转载;禁止商业使用。

关于日期转化的 文章很多.我只要讲讲我遇到的一些与日期转化有关的问题.


大约1年前,我遇到过一个很令人郁闷又费解的问题.


项目使用的是struts2框架,表单中有日期,提交表单时总是报错,说日期转化有问题,没能马上找到原因.


过了很长时间,才搞清楚关键之所在,有一个日期格式是:" yyyy-MM-dd T HH:mm:ss ".


日期和时间之间使用 T 分割,而不是采用我们熟知的 空格 来分割. " yyyy-MM-dd T HH:mm:ss " 是美国一种的日期格式,中文中是没有这种日期格式的.


我当时使用的浏览器是IE,碰巧当时IE浏览器的语言被设置成了英语,所以提交表单时struts2按照 yyyy-MM-dd T HH: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


Struts2中转换Date类型的问题


这里我要强调一点:我之所以写converter是为了 根本上解决问题 ,这样不管浏览器是什么语言都不用担心了.


我推荐使用org.apache.commons.lang3.time.DateUtils 来进行日期转化,为什么呢?


我们看到org.apache.commons.lang3.time.DateUtils 的第二个参数是一个数组,意思就是我循环的从数组中拿出一个pattern进行parse,直到解析成功为止.其实这就是一种 容错 的思维,非常好的思维.


后来在spring MVC中也遇到了关于日期转化的问题


场景:更新的时候,需要把日期放到表单的隐藏域中,但是如果日期为空的话,提交就报错,大概意思:


can not convert String "" to Date.根本原因是隐藏域的值为空.


所以解决方法:也是写一个转化器,如果日期字符串为空,则返回空就完了,不必报错.


0 0