将XMLGregorianCalendar转换为GregorianCalendar的正确方法

时间:2011-11-17 11:55:37

标签: java calendar timezone timezone-offset

我有2节课。首先包含Calendar字段和Integer字段(tz offset)。其次包含XmlGregorianCalendar字段。我想比较从第一节到第二节的日期。

Calendar cal1 = (Calendar) SerializationUtils.clone(firstClass.getDepartureDatetime());
cal1.add(Calendar.MINUTE, -firstClass.getDepartureTzOffset());

GregorianCalendar cal2 = secondClass.getDepartureDateTime().toGregorianCalendar();
cal2.add(Calendar.MINUTE, -secondClass.getDepartureDateTime().getTimezone());

if (LOGGER.isDebugEnabled()) {
                LOGGER.debug(" - Second  [" + DateFormat.getDateTimeInstance().format(cal2.getTime()) + "]");
                LOGGER.debug(" - First [" + DateFormat.getDateTimeInstance().format(cal1.getTime()) + "]");
}

我已经在这些课程中设定了相同的日期(11月19日,格林尼治标准时间上午9:00 + 1)。

根据系统TZ,它显示不同的结果(格林威治标准时间TZ):

Debian Lenny,TZ是CET:

Second  [Nov 19, 2011 7:00:00 AM] - wrong!
First [Nov 19, 2011 8:00:00 AM] -right!

Win7,TZ是GMT + 3:

Second  [Nov 19, 2011 8:30:00 AM] - wrong!
First [Nov 19, 2011 8:00:00 AM] -right!

我做错了什么?

感谢。

更新

第1和第2课:

public class FirstClass implements Serializable {
    private static final long serialVersionUID = -1150341618306402800L;

    private Calendar departureDatetime;

    private Integer departureTzOffset;

    public Calendar getDepartureDatetime() {
        return departureDatetime;
    }

    public void setDepartureDatetime(Calendar departureDatetime) {
        this.departureDatetime = departureDatetime;
    }

    public Integer getDepartureTzOffset() {
        return departureTzOffset;
    }

    public void setDepartureTzOffset(Integer departureTzOffset) {
        this.departureTzOffset = departureTzOffset;
    }
}

public class SecondClass implements Serializable
{

    private final static long serialVersionUID = 12345L;

    protected XMLGregorianCalendar departureDateTime;

    public XMLGregorianCalendar getDepartureDateTime() {
        return departureDateTime;
    }

    public void setDepartureDateTime(XMLGregorianCalendar value) {
        this.departureDateTime = value;
    }
}

SerializationUtils是来自Apache commons-lang lib的org.apache.commons.lang.SerializationUtils。

2 个答案:

答案 0 :(得分:1)

当我们调用(Calendar)SerializationUtils.clone(firstClass.getDepartureDatetime())时,Timezone会出现问题。时区设置为服务器TZ,我们在比较期间丢失了几个小时。

答案 1 :(得分:0)

你要问自己的第一个问题:我想做什么?转换GregorianCalendar和XMLGregorianCalendar非常简单:

GregorianCalendar gc;
XMLGregorianCalendar xc;
gc = xc.toGregorianCalendar();
xc = DatatypeFactory.newInstance().newXMLGregorianCalendar(gc);

但这似乎不是你问题的核心。您是否尝试执行时区转换?如果你将转换转换到显示时间(因为它确实是一个格式问题),恕我直言可以更容易地完成,利用GregorianCalendar和XMLGregorianCalendar携带他们的时区信息并摆脱两个辅助类的事实

TimeZone cet = TimeZone.getTimeZone("CET");
TimeZone utc = TimeZone.getTimeZone("UTC");
GregorianCalendar gc = new GregorianCalendar();
DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss Z");

@Test
public void testNow() throws DatatypeConfigurationException {
  df.setTimeZone(gc.getTimeZone());
  log.info(" - Gregorian LOCAL [" + df.format(gc.getTime()) + "]");
  df.setTimeZone(cet);
  log.info(" - Gregorian CET [" + df.format(gc.getTime()) + "]");
  df.setTimeZone(utc);
  String gcs = df.format(gc.getTime());
  log.info(" - Gregorian UTC [" + df.format(gc.getTime()) + "]");
  XMLGregorianCalendar xc = DatatypeFactory.newInstance().newXMLGregorianCalendar(gc);
  df.setTimeZone(xc.getTimeZone(0));
  log.info(" - XML RAW [" + df.format(xc.toGregorianCalendar().getTime()) + "]");
  df.setTimeZone(cet);
  log.info(" - XML CET [" + df.format(xc.toGregorianCalendar().getTime()) + "]");
  df.setTimeZone(utc);
  String xcs = df.format(xc.toGregorianCalendar().getTime());
  log.info(" - XML UTC [" + df.format(xc.toGregorianCalendar().getTime()) + "]");
  assertEquals(gcs, xcs);
}

或许你的问题确实是一个消毒输入的问题。我看到你有一个离开时间变量,大概是来自世界各地机场的航班,你可能从一些没有明确时区信息的数据源获得它们,而是假设“机场当地时间”。这可以解释帮助程序类,但在这种情况下,您应该在发生的情况下清理输入。确定“在机场的当地时间”有时可能很棘手(一个国家可能选择在明年晚些时候从夏令时转为标准,或者完全取消DST,并且机场甚至可能在美国转换时区,例如,从东部到中部和后部的县比你想象的更频繁地发生。您应该使用计算机的Locale数据库来解决这个问题,并避免尝试使用自己的时区算法。