4

自 Java 9 以来,Java 日期和时间 API 发生了变化。LocalDateTime 现在具有微秒精度。

Java 9 有一个全新的 java.time.Clock 实现,能够以比毫秒(小数点的三位数字)更精细的分辨率捕获当前时刻。

我们从后端服务获取时间(以微秒为单位)。

System.currentTimeMillis  > 1565245051795    > 2019-08-08T06:17:31.795
Service.getTime           > 1565245051795306 > 2019-08-08T06:17:31.795306

为了构造一个在我们的应用程序中使用的 LocalDateTime,我们做

long timeMicros = service.getTime();
long timeMillis = timeMicros / 1000;
LocalDateTime ldt = Instant.ofEpochMilli(timeMillis).atZone(ZoneId.systemDefault()).toLocalDateTime();

为了查询服务,我们再次需要时间微秒,然后我们做

long timeMillis = dateTime.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli();
long timeMicros = timeMillis * 1000;

问题是我们无法恢复时间微秒精度。

是否可以创建具有微秒精度的 Instant?
我们现在使用的是 Java 11。当我们的一个 JUnit 测试由于微秒精度的提高而失败时,我注意到了这一变化。

对于 JUnit 测试,我找到了一种解决方法:

private static final LocalDateTime START = LocalDateTime.now().truncatedTo(ChronoUnit.MILLIS);


我不确定这是一种解决方法还是一个实际的解决方案,但从时间戳添加最后三微秒数字似乎有效。

long micros = 306L; //TODO get the last three digits from the timeMicros
ldt.plus(micros, ChronoUnit.MICROS));
4

1 回答 1

7
    long timeMicros = 1_565_245_051_795_306L;
    Instant i = Instant.EPOCH.plus(timeMicros, ChronoUnit.MICROS);
    System.out.println(i);

输出是:

2019-08-08T06:17:31.795306Z

编辑:我更喜欢使用对微秒的内置支持,而不是通过除法和乘法将微秒转换为毫秒和/或秒。此外,当明确地将它们添加到时代时,感觉有点手持。

您已经知道如何转换InstantLocalDateTime,您已经在问题中展示了它,所以我不再重复。

编辑:

您有解决方案从 Instant 中恢复 timeMicros 吗?

有几个选项。这样计算就不那么复杂了,所以我可能会这样做:

    long microsBack = TimeUnit.SECONDS.toMicros(i.getEpochSecond())
            + TimeUnit.NANOSECONDS.toMicros(i.getNano());
    System.out.println(microsBack);

1565245051795306

为了使第一次转换更具风格,您可能更喜欢稍短的:

    long microsBack = ChronoUnit.MICROS.between(Instant.EPOCH, i);

编辑:可能是挑剔,但也是为了避免任何人的误解:LocalDateTime一直有纳秒精度。只有该now方法在 Java 8 上具有毫秒精度。我在某处读到 Java 9 的精度因平台而异,但您是对的,微秒精度似乎很典型。

于 2019-08-08T19:08:24.300 回答