Java 1.6 夏令时

标签 java date timezone dst

我正在开发一个关心夏令时变化的关键应用程序。
我试图通过比较跨越夏令时变化的日期来手动模拟运行时可能发生的情况,因此我进行了以下测试。 我当前所在的位置是意大利,因此今年从 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/

相关文章:

Java - 找不到主类?

java - 如何按今天/本周/本月对日期数组进行排序 - android

GridLayout 中的 Java FlowLayout

java - SimpleDateFormat 似乎忽略了时区

.net - .NET 有时区变化的历史吗?

java - 如何检查页面上是否出现了一些文本数据

java - 一般调试日志实践

java - 生成整数并丢弃最低值

c# - 查找两个时区之间的所有时区偏移周期

不同时区的Java日历对同一时间给出不同的值