为什么是ZoneOffset.UTC!= ZoneId.of(“ UTC”)?


125

为什么

ZonedDateTime now = ZonedDateTime.now();
System.out.println(now.withZoneSameInstant(ZoneOffset.UTC)
        .equals(now.withZoneSameInstant(ZoneId.of("UTC"))));

打印出来false

我希望两个ZonedDateTime实例都相等。

Answers:


180

答案来自(强调我的)javadocZoneId ...

ZoneId用于标识用于在Instant和LocalDateTime之间进行转换的规则。ID有两种不同的类型:

  • 固定偏移量-与UTC /格林威治标准时间完全抵消的偏移量,所有本地日期时间都使用相同的偏移量
  • 地理区域-适用于从UTC /格林威治中查找偏移量的一组特定规则的区域

大多数固定偏移量由ZoneOffset表示。在任何ZoneId上调用normalized()将确保将固定的偏移量ID表示为ZoneOffset。

...并且来自(强调我的)的javadocZoneId#of

此方法解析ID,生成ZoneId或ZoneOffset。如果ID为'Z'或以'+'或'-'开头,则返回ZoneOffset

参数id指定为"UTC",因此它将返回ZoneId带有偏移量的a ,该偏移量也以字符串形式显示:

System.out.println(now.withZoneSameInstant(ZoneOffset.UTC));
System.out.println(now.withZoneSameInstant(ZoneId.of("UTC")));

输出:

2017-03-10T08:06:28.045Z
2017-03-10T08:06:28.045Z[UTC]

使用equals比较方法时,将检查对象是否相等。由于存在上述差异,因此评估结果为false

当按照normalized()文档中的建议使用方法时,使用的比较equals将返回truenormalized()并将返回对应的ZoneOffset

标准化时区ID,并在可能的情况下返回ZoneOffset。

now.withZoneSameInstant(ZoneOffset.UTC)
    .equals(now.withZoneSameInstant(ZoneId.of("UTC").normalized())); // true

如文档所述,如果您使用"Z""+0"作为输入ID,of则将ZoneOffset直接返回,而无需调用normalized()

now.withZoneSameInstant(ZoneOffset.UTC).equals(now.withZoneSameInstant(ZoneId.of("Z"))); //true
now.withZoneSameInstant(ZoneOffset.UTC).equals(now.withZoneSameInstant(ZoneId.of("+0"))); //true

要检查它们是否存储相同的日期时间,可以改用isEqual方法:

now.withZoneSameInstant(ZoneOffset.UTC)
    .isEqual(now.withZoneSameInstant(ZoneId.of("UTC"))); // true

样品

System.out.println("equals - ZoneId.of(\"UTC\"): " + nowZoneOffset
        .equals(now.withZoneSameInstant(ZoneId.of("UTC"))));
System.out.println("equals - ZoneId.of(\"UTC\").normalized(): " + nowZoneOffset
        .equals(now.withZoneSameInstant(ZoneId.of("UTC").normalized())));
System.out.println("equals - ZoneId.of(\"Z\"): " + nowZoneOffset
        .equals(now.withZoneSameInstant(ZoneId.of("Z"))));
System.out.println("equals - ZoneId.of(\"+0\"): " + nowZoneOffset
        .equals(now.withZoneSameInstant(ZoneId.of("+0"))));
System.out.println("isEqual - ZoneId.of(\"UTC\"): "+ nowZoneOffset
        .isEqual(now.withZoneSameInstant(ZoneId.of("UTC"))));

输出:

equals - ZoneId.of("UTC"): false
equals - ZoneId.of("UTC").normalized(): true
equals - ZoneId.of("Z"): true
equals - ZoneId.of("+0"): true
isEqual - ZoneId.of("UTC"): true

4
文档还说:“如果区域ID等于'GMT','UTC'或'UT',则结果是具有相同ID的ZoneId和等效于ZoneOffset.UTC的规则”。相同的ID和规则,但行为不同。ZoneId.of("Z")给您ZoneOffset.UTCZoneId.of("UTC")给您一个ZoneId(不是ZoneOffset.UTC)。至少可以说,此API是不直观的。
亚当·米勒奇普
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.