我有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。
答案 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数据库来解决这个问题,并避免尝试使用自己的时区算法。