您问题中的第二个输出是运行爱尔兰时间(欧洲/都柏林)的 JVM 上的正确和预期行为。2017 年 9 月 12 日,爱尔兰处于夏令时 (DST)。虽然没有明确记录, (您在打印时Date.toString()
隐式调用)在 JVM 的时区中打印日期和时间,在 9 月呈现为爱尔兰夏令时的 IST。Date
c.getTime()
当您Calendar
也使用爱尔兰时间在对象上设置日期时,将保留一天中的小时;在您的情况下,您将获得 2007 年 1 月 1 日 12:36:24 爱尔兰标准时间。现在想象一下,如果爱尔兰夏令时间和爱尔兰标准时间都被呈现为 IST,会造成什么混乱。你将无法区分。相反,由于爱尔兰标准时间与格林威治标准时间重合,因此Date.toString()
当日期不在一年中的夏季时间(1 月不在)时打印。
我的猜测是您的第一个输出来自运行印度时间的 JVM。它也被呈现为 IST,并且由于印度不使用夏令时,因此夏季和冬季使用相同的缩写。
java.time
在理解您观察到的行为的解释之前,我发表了关于过时和现代 Java 日期和时间类的评论。不过,我仍然不认为评论已经过时了。这是您的代码的现代等价物:
ZonedDateTime zdt = ZonedDateTime.now(ZoneId.of("Europe/Dublin"));
System.out.println(zdt);
zdt = zdt.with(LocalDate.of(2007, Month.JANUARY, 1));
System.out.println(zdt);
它打印
2017-09-12T11:45:33.921+01:00[Europe/Dublin]
2007-01-01T11:45:33.921Z[Europe/Dublin]
如果要使用 JVM 的时区设置,请ZoneId.systemDefault()
使用ZoneId.of("Europe/Dublin")
. 正如名称所述,与 相反Date
,ZonedDateTime
确实包括时区。它更多地对应于旧Calendar
类。如您所见,它的方法以明确的区域/城市格式toString
打印与 UTC 的偏移量(Z
意味着零偏移量)和时区名称。我相信这会减少混乱的空间。如果要以特定格式打印日期,请使用.DateTimeFormatter
附录:代码的示例输出
为了完整起见,以下是您的代码在运行可能呈现为 IST 的不同时区时的输出:
欧洲/都柏林(同意你的第二个输出)
Tue Sep 12 11:19:28 IST 2017
Mon Jan 01 11:19:28 GMT 2007
亚洲/特拉维夫
Tue Sep 12 13:19:28 IDT 2017
Mon Jan 01 13:19:28 IST 2007
亚洲/加尔各答(同意你的第一个输出)
Tue Sep 12 15:49:28 IST 2017
Mon Jan 01 15:49:28 IST 2007