Java 8 有一个全新的日期和时间 API。此 API 中最有用的类之一是 LocalDateTime
,用于保存与时区无关的日期与时间值。
可能有数百万行代码使用遗留类 java.util.Date
用于此目的。因此,在连接新旧代码时,需要在两者之间进行转换。由于似乎没有直接的方法可以做到这一点,如何做到这一点?
简答:
Date in = new Date();
LocalDateTime ldt = LocalDateTime.ofInstant(in.toInstant(), ZoneId.systemDefault());
Date out = Date.from(ldt.atZone(ZoneId.systemDefault()).toInstant());
说明:(基于this question 关于LocalDate
)
尽管有它的名字,java.util.Date
代表时间线上的一个瞬间,而不是“日期”。对象中存储的实际数据是自 1970-01-01T00:00Z(1970 GMT/UTC 开始时的午夜)以来的 long
毫秒计数。
JSR-310中java.util.Date
的等价类是Instant
,因此有方便的方法来提供来回转换:
Date input = new Date();
Instant instant = input.toInstant();
Date output = Date.from(instant);
java.util.Date
实例没有时区的概念。如果您在 java.util.Date
上调用 toString()
,这可能看起来很奇怪,因为 toString
是相对于时区的。然而,该方法实际上使用 Java 的默认时区来提供字符串。时区不是 java.util.Date
实际状态的一部分。
Instant
也不包含任何有关时区的信息。因此,要将 Instant
转换为本地日期时间,必须指定时区。这可能是默认区域 - ZoneId.systemDefault()
- 也可能是您的应用程序控制的时区,例如用户偏好设置的时区。 LocalDateTime
有一个方便的工厂方法,它同时采用即时和时区:
Date in = new Date();
LocalDateTime ldt = LocalDateTime.ofInstant(in.toInstant(), ZoneId.systemDefault());
反过来,LocalDateTime
时区是通过调用 atZone(ZoneId)
方法来指定的。 ZonedDateTime
然后可以直接转换为 Instant
:
LocalDateTime ldt = ...
ZonedDateTime zdt = ldt.atZone(ZoneId.systemDefault());
Date output = Date.from(zdt.toInstant());
请注意,从 LocalDateTime
到 ZonedDateTime
的转换可能会引入意外行为。这是因为由于夏令时,并非每个本地日期时间都存在。在秋季/秋季,本地时间线存在重叠,相同的本地日期时间出现两次。在 Spring ,有一个间隙,一个小时消失了。参见 atZone(ZoneId)
的 Javadoc有关转换将做什么的更多定义。
总结,如果您将 java.util.Date
往返到 LocalDateTime
并返回到 java.util.Date
您由于夏令时,可能会以不同的瞬间结束。
附加信息:还有一个差异会影响非常旧的日期。 java.util.Date
使用在 1582 年 10 月 15 日更改的日历,在此之前的日期使用儒略历而不是公历。相比之下,java.time.*
一直使用 ISO 日历系统(相当于公历)。在大多数用例中,您需要 ISO 日历系统,但在比较 1582 年之前的日期时,您可能会看到奇怪的效果。
我是一名优秀的程序员,十分优秀!