如何使用Java中的TimeZone解决夏令时问题

Sur*_*dra 58 java timezone calendar date

我必须在我的Java应用程序中打印EST时间.我使用以下方法将时区设置为EST:

Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("EST"));
Run Code Online (Sandbox Code Playgroud)

但是当在这个时区中遵循夏令时时,我的代码不能打印正确的时间(它打印的时间减少1小时).

无论是否观察到夏令时,如何使代码始终能够正确读取正确的时间?

PS:我尝试将时区设置为EDT,但它没有解决问题.

Jon*_*eet 106

这是问题的开始:

Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("EST"));
Run Code Online (Sandbox Code Playgroud)

应全心全意地避免使用3个字母的缩写,以支持TZDB区域ID.EST是东部标准时间 - 标准时间从未观察到DST; 它不是真正的全时区名称.它是用于部分时区的名称.(不幸的是,我没有遇到这个"半时区"概念的好词.)

您需要一个完整的时区名称.例如,America/New_York在东部时区:

TimeZone zone = TimeZone.getTimeZone("America/New_York");
DateFormat format = DateFormat.getDateTimeInstance();
format.setTimeZone(zone);

System.out.println(format.format(new Date()));
Run Code Online (Sandbox Code Playgroud)


Bas*_*que 17

其他答案是正确的,尤其是Jon Skeet的答案,但过时了.

java.time

这些旧的日期时间类已被Java 8及更高版本中内置的java.time框架所取代.

如果您只想要UTC中的当前时间,请使用Instant该类.

Instant now = Instant.now();
Run Code Online (Sandbox Code Playgroud)

EST不是时区,正如Jon Skeet正确答案中所解释的那样.这样的3-4个字母代码既不是标准化也不是唯一的,并且进一步混淆了夏令时(DST).使用"continent/region"格式的正确时区名称.

也许你的意思是北美东海岸的东部标准时间?还是埃及标准时间?还是欧洲标准时间?

ZoneId zoneId = ZoneId.of( "America/New_York" );
ZoneId zoneId = ZoneId.of( "Africa/Cairo" );
ZoneId zoneId = ZoneId.of( "Europe/Lisbon" );
Run Code Online (Sandbox Code Playgroud)

使用任何此类ZoneId对象将当前时刻调整到特定时区以生成ZonedDateTime对象.

ZonedDateTime zdt = ZonedDateTime.now( zoneId ) ;
Run Code Online (Sandbox Code Playgroud)

通过从第一个生成另一个ZonedDateTime对象,将ZonedDateTime调整到不同的时区.java.time框架使用不可变对象而不是更改(变异)现有对象.

ZonedDateTime zdtGuam = zdt.withZoneSameInstant( ZoneId.of( "Pacific/Guam" ) ) ;
Run Code Online (Sandbox Code Playgroud)

  • 添加秒确实会在日期和时间方面停止计时.我们需要额外的时间才能进入日历中的下一个日期.所以我们的日历吸收了闰秒.正如我所说,这是闰秒的全部要点:减慢我们的时钟以减缓我们的日历跟踪.虽然你忽略闰秒的观点在技术上在java.time类的内部表示方面是正确的,但实际上你的观点是无关紧要的,迂腐的,并且对于努力学习如何表示日期和时间的许多程序员来说是不必要的混淆. (2认同)