我正在开发一个关心夏令时变化的关键应用程序。
我试图通过比较跨越夏令时变化的日期来手动模拟运行时可能发生的情况,因此我进行了以下测试。
我当前所在的位置是意大利,因此今年从 CEST(中欧夏令时)更改为 CET(中欧时间)发生在 25/10。
我用了full time zone names ,我的时区是 Europe/Rome
。
这是我做的测试:
Calendar before = Calendar.getInstance(TimeZone.getTimeZone("Europe/Rome DST")); //CEST
before.set(Calendar.DAY_OF_MONTH, 25);
before.set(Calendar.MONTH, Calendar.OCTOBER);
before.set(Calendar.HOUR_OF_DAY, 2);
before.set(Calendar.MINUTE, 30);
before.set(Calendar.SECOND, 0);
before.set(Calendar.MILLISECOND, 0);
System.out.println(before.getTime());
Calendar after = Calendar.getInstance(TimeZone.getTimeZone("Europe/Rome")); //CET
after.set(Calendar.DAY_OF_MONTH, 25);
after.set(Calendar.MONTH, Calendar.OCTOBER);
after.set(Calendar.HOUR_OF_DAY, 2);
after.set(Calendar.MINUTE, 30);
after.set(Calendar.SECOND, 0);
after.set(Calendar.MILLISECOND, 0);
System.out.println(after.getTime());
System.out.println(before.compareTo(after));
输出是:
BEFORE DST CHANGE: Sun Oct 25 03:30:00 CET 2015
AFTER DST CHANGE: Sun Oct 25 02:30:00 CET 2015
before.compareTo(after): 1
比较结果是错误的,即 2:30 CEST 在 2:30 CET 之后,但恰恰相反。
不知道是不是真考。
有什么办法可以解决这个问题吗?
我也尝试了 joda 时间,但结果是一样的。
提前致谢。
最佳答案
您的问题是 "Europe/Rome DST"
无法被 getTimeZone(timeZoneId)
识别。
当它不理解您的输入时,it returns the GMT timezone by default .您可以使用 getAvailableIDs
(上述链接中 getTimeZone
下方的方法)查看可用时区 ID 的列表。
需要注意的是CEST
也不在列表中。要模拟 CEST
时区,您可以选择以下解决方案之一:
- 我建议使用
TimeZone.setRawOffset(int offsetInMs)
自己设置 CET 和 CEST 的偏移量。 - 使用相对于 GMT 定义的时区之一(例如,id
"Etc/GMT+1"
)。这将确保您使用的是 TimeZone api 能够理解的有效时区偏移量。 - 在日历实例上设置 DST 偏移量
Calendar.DST_OFFSET
.
通过使用最后一个解决方案,正确的测试代码是:
Calendar before = Calendar.getInstance(TimeZone.getTimeZone("Europe/Rome"));
before.set(Calendar.DST_OFFSET, 3600000);
before.set(Calendar.DAY_OF_MONTH, 25);
before.set(Calendar.MONTH, Calendar.OCTOBER);
before.set(Calendar.HOUR_OF_DAY, 2);
before.set(Calendar.MINUTE, 30);
before.set(Calendar.SECOND, 0);
before.set(Calendar.MILLISECOND, 0);
System.out.println("BEFORE DST CHANGE: " + before.getTime());
Calendar after = Calendar.getInstance(TimeZone.getTimeZone("Europe/Rome"));
after.set(Calendar.DAY_OF_MONTH, 25);
after.set(Calendar.MONTH, Calendar.OCTOBER);
after.set(Calendar.HOUR_OF_DAY, 2);
after.set(Calendar.MINUTE, 30);
after.set(Calendar.SECOND, 0);
after.set(Calendar.MILLISECOND, 0);
System.out.println("AFTER DST CHANGE: " + after.getTime());
System.out.println("before.compareTo(after): " + before.compareTo(after));
输出:
BEFORE DST CHANGE: Sun Oct 25 02:30:00 CEST 2015
AFTER DST CHANGE: Sun Oct 25 02:30:00 CET 2015
before.compareTo(after): -1
关于Java 1.6 夏令时,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30442031/