使用 Instant/ChronoUnit 获取今天午夜的 nanos-since-epoch 时间比使用 GregorianCalendar/Date 提前 24 小时

Posted

技术标签:

【中文标题】使用 Instant/ChronoUnit 获取今天午夜的 nanos-since-epoch 时间比使用 GregorianCalendar/Date 提前 24 小时【英文标题】:Using Instant/ChronoUnit to get the time in nanos-since-epoch at midnight today is 24 hours ahead of using GregorianCalendar/Date 【发布时间】:2021-05-17 09:06:09 【问题描述】:

我正在尝试从 Epoch 午夜(例如 2021 年 2 月 13 日 00:00:00)开始以 nanos 形式获取今天的日期。

使用 GregorianCalendar/Date 似乎可以得到正确的结果。

我使用 Instant/ChronoUnit 的方式是明天午夜给我(例如 2021 年 2 月 14 日 00:00:00)。

这里有什么问题?

import java.time.Instant;
import java.time.temporal.ChronoUnit;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.Calendar;
import java.util.TimeZone;


class scratch 

    public static void main(String[] args) 
        Calendar c = new GregorianCalendar();
        c.setTime(new Date());
        c.set(Calendar.HOUR_OF_DAY, 0);
        c.set(Calendar.MINUTE, 0);
        c.set(Calendar.SECOND, 0);
        c.set(Calendar.MILLISECOND, 0);
        c.setTimeZone(TimeZone.getTimeZone("UTC"));
        long dateNanos = c.getTimeInMillis() * 1000000L;

        Instant i = Instant.now().truncatedTo(ChronoUnit.DAYS);
        long instantNanos = ChronoUnit.NANOS.between(Instant.EPOCH, i);

        System.out.println("dateMillis = " + dateNanos);
        System.out.println("instantMillis = " + instantNanos);
        System.out.println("instantMillis - dateMillis = " + (instantNanos - dateNanos));
    

trucatedTo 似乎正在四舍五入。 javadocs 说它四舍五入。

例如,使用 MINUTES 单位截断将向下舍入到最接近的分钟,将秒和纳秒设置为零。

【问题讨论】:

【参考方案1】:

这些会给出不同的结果,因为您在不同的时区“截断到当天”。请注意,当您在不同的时区时,“截断到当天”会给您不同的答案,因为“午夜”的时刻在不同的时区是不同的。由于 DST 更改,某些时区甚至没有时间 00:00:00。

Calendar 版本中,您将截断到系统时区中的日期。请注意,setTimeZone 调用发生在截断之后,因此它实际上并没有做任何有用的事情。

Instant 版本中,您将截断到UTC 的日期,因为Instants 中没有“时区”的概念。

Instant 只是“错误”,因为您希望它在您的系统时区中截断为一天。 Instant 并不是真的要这样做。你应该使用ZonedDateTime:

ZonedDateTime now = ZonedDateTime.now(); // this gets the current ZonedDateTime, you can also specify a specific zone
System.out.println(now.truncatedTo(ChronoUnit.DAYS).toEpochSecond() * 1_000_000_000L);

另一方面,如果您希望 Calendar 在 UTC 时区截断,请在截断之前调用 setTimeZone

Calendar c = new GregorianCalendar();
c.setTime(new Date());
c.setTimeZone(TimeZone.getTimeZone("UTC"));
c.set(Calendar.HOUR_OF_DAY, 0);
c.set(Calendar.MINUTE, 0);
c.set(Calendar.SECOND, 0);
c.set(Calendar.MILLISECOND, 0);

但我仍然建议您尽可能使用java.time :)


您现在应该做的是确定您指的是哪个“午夜”(哪个时区的午夜?)。

【讨论】:

谢谢。是的,这正是问题所在。看到这两个不同,我什至没有想到当前时间的 UTC 实际上是明天。顺便说一句,在测试您的答案时,我发现 c.setTimeZone 必须在 c.setTime 之前出现。 @marathon 真的吗?把它放在setTime 之后对我有用……我想这是不使用Calendar 的另一个原因:D

以上是关于使用 Instant/ChronoUnit 获取今天午夜的 nanos-since-epoch 时间比使用 GregorianCalendar/Date 提前 24 小时的主要内容,如果未能解决你的问题,请参考以下文章

获取今天和明天的记录,忽略周末

获取今天时间并且按照规定格式返回

如何获取产品今天和昨天价格的差异,并找出产品昨天是不是有货?

获取一年前的今天日​​期

js获取日期:昨天今天和明天后天

我需要从 mysql 数据库中获取日期并打印今天和获取日期之间的差异