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

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

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

我试图从 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 单位截断将向下舍入到最接近的分钟,将秒和纳秒设置为零。

解决方法

这些给出了不同的结果,因为您在不同的时区“截断到当天”。请注意,当您处于不同的时区时,“截断到白天”会给您不同的答案,因为“午夜”的时刻在不同的时区是不同的。由于夏令时的变化,有些时区甚至没有 00:00:00 的时间。

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

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

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 :)


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

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