java 8 - ZonedDateTime不等于另一个ZonedDateTime

L.G*_*.G. 13 java java-8 java-time

我创建了两个ZonedDateTime对象,我认为它们应该是相同的:

public static void main(String[] args) {
    ZoneId zid = ZoneId.of("America/New_York");
    ZoneOffset offset = ZoneOffset.from(LocalDateTime.now().atZone(zid));
    ZonedDateTime zdt0 = ZonedDateTime.of(2014, 8, 24, 21, 10, 1, 777000002, offset);
    ZonedDateTime zdt1 = ZonedDateTime.of(2014, 8, 24, 21, 10, 1, 777000002, zid);
    boolean equals = Objects.equals(zdt0, zdt1);
    System.out.println("equals: " + equals);
}
Run Code Online (Sandbox Code Playgroud)

在调试器中,我看到ZonedDateTime 区域的成员类在第一种情况下是java.time.ZoneOffset,在第二种情况下是java.time.ZoneRegion,这使得ZonedDateTime对象不相等.这令人困惑......有什么想法吗?

Hol*_*ger 24

您正在检查对象相等性,其评估false为这些对象不等效.一个是a ZoneId,另一个是a ZoneOffset.如果要检查它们是否代表同一时间,可以使用不是非常直观的命名方法isEqual.

例如:

ZoneId zid = ZoneId.of("America/New_York");
ZoneOffset offset = ZoneOffset.from(LocalDateTime.now().atZone(zid));
ZonedDateTime zdt0 = ZonedDateTime.of(2014, 8, 24, 21, 10, 1, 777000002, offset);
ZonedDateTime zdt1 = ZonedDateTime.of(2014, 8, 24, 21, 10, 1, 777000002, zid);
System.out.println("isEqual:" + zdt0.isEqual(zdt1));
System.out.println("equals: " + zdt0.equals(zdt1));
Run Code Online (Sandbox Code Playgroud)

打印:

isEqual:true
equals: false
Run Code Online (Sandbox Code Playgroud)

顺便说一下,请注意,您不需要使用Objects.equals(a,b)已知的两个对象null.您可以a.equals(b)直接调用.


ano*_*932 8

当使用 Jackson 序列化/反序列化 ZonedDateTime 的实例,然后将它们相互比较以验证我的代码是否正常工作时,这也让我困扰了几个小时。我不完全理解其中的含义,但我所学到的只是使用 isEqual 而不是 equals。但这会给测试计划带来很大的麻烦,因为大多数断言实用程序只会调用标准的 .equals()。

这是我在挣扎了一段时间后终于想出的:

@Test
public void zonedDateTimeCorrectlyRestoresItself() {

    // construct a new instance of ZonedDateTime
    ZonedDateTime now = ZonedDateTime.now(ZoneId.of("Z"));
    // offset = {ZoneOffset@3820} "Z"
    // zone   = {ZoneOffset@3820} "Z"

    String starting = now.toString();

    // restore an instance of ZonedDateTime from String
    ZonedDateTime restored = ZonedDateTime.parse(starting);
    // offset = {ZoneOffset@3820} "Z"
    // zone   = {ZoneOffset@3820} "Z"

    assertThat(now).isEqualTo(restored); // ALWAYS succeeds

    System.out.println("test");
}

@Test
public void jacksonIncorrectlyRestoresZonedDateTime() throws Exception {

    ObjectMapper objectMapper = new ObjectMapper();
    objectMapper.findAndRegisterModules();

    // construct a new instance of ZonedDateTime
    ZonedDateTime now = ZonedDateTime.now(ZoneId.of("Z"));
    // offset = {ZoneOffset@3820} "Z"
    // zone   = {ZoneOffset@3820} "Z"


    String converted = objectMapper.writeValueAsString(now);

    // restore an instance of ZonedDateTime from String
    ZonedDateTime restored = objectMapper.readValue(converted, ZonedDateTime.class);
    // offset = {ZoneOffset@3820} "Z"
    // zone   = {ZoneOffset@3821} "UTC"

    assertThat(now).isEqualTo(restored); // NEVER succeeds

    System.out.println("break point me");
}
Run Code Online (Sandbox Code Playgroud)