为什么在Java日历对象上设置时区没有效果?

时间:2017-12-16 03:49:38

标签: java date datetime calendar timezone

请考虑以下代码段。

    TimeZone tz = TimeZone.getTimeZone("GMT");
    System.out.println(tz);

    Calendar cal = Calendar.getInstance();
    cal.set(2017, Calendar.DECEMBER, 25);

    System.out.println("before setting time zone: " + cal.getTime());
    cal.setTimeZone(tz);

    System.out.println("after setting time zone: " + cal.getTime());

结果如下:

sun.util.calendar.ZoneInfo[id="GMT",offset=0,dstSavings=0,useDaylight=false,transitions=0,lastRule=null]
before setting time zone: Mon Dec 25 22:39:53 EST 2017 
after setting time zone: Mon Dec 25 22:39:53 EST 2017

在时区更改后,为什么日期的打印输出仍显示EST?不应该是GMT吗?

1 个答案:

答案 0 :(得分:5)

TL;博士

LocalDate.of( 2017 , Month.DECEMBER , 25 )
         .atStartOfDay( ZoneId.of( "America/New_York" ) )

避免遗留日期时间类

您正在使用现在遗留下来的非常麻烦的旧日期时间类。

要直接回答您的问题,Calendar::getTime方法会返回Date。在这些类中的许多设计缺陷中,Date::toString在生成字符串时动态应用JVM的当前默认时区。因此,您EST正在出现。

永远不要使用这些类。仅使用他们的替代品,业界领先的java.time类。

java.time

LocalDate类表示没有时间且没有时区的仅限日期的值。

LocalDate xmas2017 = LocalDate.of( 2017 , Month.DECEMBER , 25 ) ;

如果您想要日期时间,即当天的第一时刻,则必须指定时区。时区对于确定日期至关重要。对于任何给定的时刻,日期在全球范围内因地区而异。例如,在Paris France午夜后的几分钟是新的一天,而Montréal Québec中仍然是“昨天”。

continent/region的格式指定proper time zone name,例如America/MontrealAfrica/CasablancaPacific/Auckland永远不要使用3-4字母缩写,例如EST IST,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。

永远不要假设这一天从00:00开始。在某些区域,Daylight Saving Time (DST)等异常表示该日可能会在其他时间开始,例如01:00:00。让java.time确定一天中的第一个时刻。

应用ZoneId获取ZonedDateTime

ZoneId z = ZoneId.of( "America/New_York" ) ;
ZonedDateTime zdtNewYork = xmas2017.atStartOfDay( z ) ;

请注意,圣诞节在印度比在纽约早得多。

ZonedDateTime zdtKolkta = xmas2017.atStartOfDay( ZoneId.of( "Asia/Kolkata" ) ) ;

这就是为什么精灵的后勤部门将圣诞老人送往Kiribati进行首次交付。这些岛屿的最早圣诞节时区比UTC早14小时。从那里开始,当地球旋转到新的一天的阳光下时,驯鹿向东飞行,一个接一个地追逐连续的中午。

如果您希望将同一时刻调整为UTC,请提取始终为UTC的Instant

Instant instant = zdt.toInstant() ;

关于java.time

java.time框架内置于Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.DateCalendar和& SimpleDateFormat

现在位于Joda-Timemaintenance mode项目建议迁移到java.time类。

要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。规范是JSR 310

从哪里获取java.time类?

ThreeTen-Extra项目使用其他类扩展java.time。该项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如IntervalYearWeekYearQuartermore