微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

Java 8 和 Java 11 中 LocalDateTime 到毫秒的差异 为什么它适用于 Java 8?

如何解决Java 8 和 Java 11 中 LocalDateTime 到毫秒的差异 为什么它适用于 Java 8?

我目前正在将一些项目从 Java 8 升级到 Java 11,其中一个转换器的单元测试失败了。基本上问题源于由于先前通过 JDK 8 传递的日期精度导致的相等性检查失败。

这是测试的部分示例,为了清楚起见,我移动了转换器的内容

@Test
public void testDateTime() {
    LocalDateTime expected = LocalDateTime.Now().plusDays(1L);

    // converter contents
    long epochMillis = expected.atZone(ZoneId.systemDefault())
            .toInstant().toEpochMilli();
    LocalDateTime actual = LocalDateTime.ofInstant(Instant.ofEpochMilli(epochMillis),TimeZone.getDefault().toZoneId());


    assertthat(actual,equalTo(expected));
}

这会导致资产错误,原因如下:

Expected :<2021-06-02T14:06:21.820299>
Actual   :<2021-06-02T14:06:21.820>

我可以使用 assertthat(actual,equalTo(expected.truncatedTo(ChronoUnit.MILLIS))) 对预期进行中继以使它们相等,但是,这意味着每次与被测试的转换器类进行比较(isAfter、isBefore、equals)时,都必须应用中继.

对于 JDK 11(或者我可能错过的文档 :)),是否有正确的方法可以在 LocalDateTimeLong 之间进行转换,反之亦然?


更新:

正如评论中所指出的,Java 8 和 11 的表示形式不同,因此导致测试失败。为了提供有关这篇文章所要求内容的更多上下文,这里是测试验证的 2 种方法(我将其移至测试本身以仅捕获正在执行的内容,因为失败的单元测试属于使用实用方法

public Long localDateTimetoEpochMillis(LocalDateTime ldt) {
    Instant instant = ldt.atZone(ZoneId.systemDefault()).toInstant();
        return ldt.atZone(ZoneId.systemDefault())
           .toInstant().toEpochMilli();
}

public LocalDateTime epochMillisToLocalDateTime(long epochMillis) {
    return LocalDateTime.ofInstant(
            Instant.ofEpochMilli(epochMillis),ZoneId.systemDefault());
}

现有测试似乎验证的是,给定一个 long 值,我应该得到相同的 LocalDateTime 等效项,这是通过使用 Given(LocalDateTime 转换为 Long 值)然后返回 LocalDateTime 进行比较来完成的。

解决方法

如果你看一下区别:

Expected :<2021-06-02T14:06:21.820299>
Actual   :<2021-06-02T14:06:21.820>

您可以看到它删除了不到一毫秒的任何内容。

发生这种情况是因为您将 LocalDateTime 转换为毫秒:

.toInstant().toEpochMilli();

为了避免这种情况,您可以使用 Instant#getNano:

获取从秒开始沿时间线稍后的纳秒数。 纳秒秒值测量从 getEpochSecond() 返回的秒算起的总纳秒数。

它可能看起来像这样:

Instant instant=expected.atZone(ZoneId.systemDefault())
                .toInstant();
long epochMillis = instant.toEpochMilli();
long nanos=instant.getNano()%1000000;//get nanos of Millisecond

LocalDateTime actual = LocalDateTime.ofInstant(Instant.ofEpochMilli(epochMillis).plusNanos(nanos),TimeZone.getDefault().toZoneId());

为什么它适用于 Java 8?

正如 this postJDK-8068730 Increase the precision of the implementation of java.time.Clock.systemUTC() 所描述的,Java 8 没有捕获小于一毫秒的时间单位。从 Java 9 开始,LocalDateTime.now(和类似的)以微秒为单位获取时间。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。