在java.time.LocalDateTime和java.util.Date之间转换

Knu*_*daa 456 java datetime java-8 java-time

Java 8有一个全新的日期和时间API.此API中最有用的类之一是LocalDateTime,用于保存与时区无关的日期时间值.

java.util.Date为此目的,使用遗留类可能有数百万行代码.因此,当连接新旧代码时,需要在两者之间进行转换.由于似乎没有直接的方法来实现这一点,怎么办呢?

Jod*_*hen 679

简短回答:

Date in = new Date();
LocalDateTime ldt = LocalDateTime.ofInstant(in.toInstant(), ZoneId.systemDefault());
Date out = Date.from(ldt.atZone(ZoneId.systemDefault()).toInstant());
Run Code Online (Sandbox Code Playgroud)

说明:(基于这个问题有关LocalDate)

尽管它的名字,java.util.Date代表了时间线上的瞬间,而不是"日期".存储在对象中的实际数据是long自1970-01-01T00:00Z(格林威治标准时间1970 /格林威治标准时间1970年初午夜)以来的毫秒数.

java.util.Date在JSR-310中的等价类是Instant,因此有方便的方法来提供转换:

Date input = new Date();
Instant instant = input.toInstant();
Date output = Date.from(instant);
Run Code Online (Sandbox Code Playgroud)

一个java.util.Date实例没有时区的概念.如果你打电话toString()给a java.util.Date,这可能看起来很奇怪,因为它toString是相对于时区的.但是,该方法实际上使用Java的默认时区来提供字符串.时区不是实际状态的一部分java.util.Date.

An Instant也不包含有关时区的任何信息.因此,要从Instant本地日期时间转换为本地日期时间,必须指定时区.这可能是默认区域 - ZoneId.systemDefault()或者它可能是应用程序控制的时区,例如来自用户首选项的时区.LocalDateTime有一个方便的工厂方法,既采用即时和时区:

Date in = new Date();
LocalDateTime ldt = LocalDateTime.ofInstant(in.toInstant(), ZoneId.systemDefault());
Run Code Online (Sandbox Code Playgroud)

反之,LocalDateTime通过调用atZone(ZoneId)方法指定时区.所述ZonedDateTime然后可以直接转化为Instant:

LocalDateTime ldt = ...
ZonedDateTime zdt = ldt.atZone(ZoneId.systemDefault());
Date output = Date.from(zdt.toInstant());
Run Code Online (Sandbox Code Playgroud)

请注意,从转换LocalDateTimeZonedDateTime有可能引入意外行为.这是因为夏令时不会存在每个本地日期时间.在秋季/秋季,当地时间线存在重叠,其中相同的本地日期时间出现两次.在春天,有一个缺口,一小时消失.有关atZone(ZoneId)转换将执行的更多定义,请参阅Javadoc .

总结一下,如果你往返a java.util.Date到a LocalDateTime并回到a java.util.Date,由于夏令时,你可能会得到一个不同的瞬间.

附加信息:还有另一个区别会影响很久的日期.java.util.Date使用日历在1582年10月15日更改,日期之前使用儒略历而不是格里高利日历.相比之下,java.time.*始终使用ISO日历系统(相当于格里高利).在大多数用例中,ISO日历系统是您想要的,但在比较1582年之前的日期时,您可能会看到奇怪的效果.

  • 我添加了一个关于非常古老的日期(1582 年之前)的部分。FWIW,您建议的修复很可能是错误的,因为 java.util.Date 中的 1111-11-11 与 java.time 中的 1111-11-18 在历史上是同一天,因为日历系统不同(6.5 分钟的差异)出现在 1900 年之前的许多时区) (3认同)
  • 非常感谢您的明确解释.特别是为什么`java.util.Date`不包含时区,但在`toString()`期间打印它.事件[官方文档](http://docs.oracle.com/javase/6/docs/api/java/util/Date.html#toString())在发布时没有明确说明. (2认同)
  • 还值得注意的是,`java.sql.Date#toInstant` 会抛出一个 `UnsupportedOperationException`。所以不要在 `java.sql.ResultSet#getDate` 的 RowMapper 中使用 `toInstant`。 (2认同)

The*_*tor 126

这就是我想出来的(和所有日期时间难题一样,它可能会因为一些奇怪的时区 - 白昼 - 日光调整而被反驳:D)

往返: Date<< - >>LocalDateTime

鉴于: Date date = [some date]

(1)LocalDateTime<< Instant<<Date

    Instant instant = Instant.ofEpochMilli(date.getTime());
    LocalDateTime ldt = LocalDateTime.ofInstant(instant, ZoneOffset.UTC);
Run Code Online (Sandbox Code Playgroud)

(2) Date<< Instant<<LocalDateTime

    Instant instant = ldt.toInstant(ZoneOffset.UTC);
    Date date = Date.from(instant);
Run Code Online (Sandbox Code Playgroud)

例:

鉴于:

Date date = new Date();
System.out.println(date + " long: " + date.getTime());
Run Code Online (Sandbox Code Playgroud)

(1)LocalDateTime<< Instant<< Date:

创建InstantDate:

Instant instant = Instant.ofEpochMilli(date.getTime());
System.out.println("Instant from Date:\n" + instant);
Run Code Online (Sandbox Code Playgroud)

创建DateInstant(不是必须的,但为了举例说明):

date = Date.from(instant);
System.out.println("Date from Instant:\n" + date + " long: " + date.getTime());
Run Code Online (Sandbox Code Playgroud)

LocalDateTime从中创建Instant

LocalDateTime ldt = LocalDateTime.ofInstant(instant, ZoneOffset.UTC);
System.out.println("LocalDateTime from Instant:\n" + ldt);
Run Code Online (Sandbox Code Playgroud)

(2)Date<< Instant<<LocalDateTime

创建InstantLocalDateTime:

instant = ldt.toInstant(ZoneOffset.UTC);
System.out.println("Instant from LocalDateTime:\n" + instant);
Run Code Online (Sandbox Code Playgroud)

创建DateInstant:

date = Date.from(instant);
System.out.println("Date from Instant:\n" + date + " long: " + date.getTime());
Run Code Online (Sandbox Code Playgroud)

输出是:

Fri Nov 01 07:13:04 PDT 2013 long: 1383315184574

Instant from Date:
2013-11-01T14:13:04.574Z

Date from Instant:
Fri Nov 01 07:13:04 PDT 2013 long: 1383315184574

LocalDateTime from Instant:
2013-11-01T14:13:04.574

Instant from LocalDateTime:
2013-11-01T14:13:04.574Z

Date from Instant:
Fri Nov 01 07:13:04 PDT 2013 long: 1383315184574
Run Code Online (Sandbox Code Playgroud)

  • 作为一般的JSR-310规则,不需要使用epoch-millis在类型之间进行转换.使用对象存在更好的选择,请参阅下面的完整答案.如果使用像UTC这样的区域偏移,上面的答案也是完全有效的 - 答案的某些部分不适用于像America/New_York这样的完整时区. (5认同)
  • @scottb为什么第三方案件不常见?它该死的常见.一个例子:JDBC 4和更少(希望不是5). (3认同)
  • @scottb你是在向我或提出问题的人发表这个问题吗?至于我的想法,那么在jdk 8 API中明确说明转换将是很好的** - 他们至少可以做到这一点.无论如何,有许多库将被重新考虑以包含新的Java-8功能,并且知道如何做到这一点非常重要. (2认同)
  • 而不是`Instant.ofEpochMilli(date.getTime())`do`date.toInstant()` (2认同)

小智 19

如果您确定需要默认时区,那么更方便的方法:

Date d = java.sql.Timestamp.valueOf( myLocalDateTime );
Run Code Online (Sandbox Code Playgroud)

  • 当然,它更容易,但我不喜欢混合jdbc相关的东西与简单的日期处理,至少恕我直言. (21认同)
  • 对不起,这是解决一个简单问题的可怕办法.这就像定义5等于奥林匹克标志中的戒指数量. (6认同)

Luk*_*ski 11

LocalDateTime->的最快方法Date是:

Date.from(ldt.toInstant(ZoneOffset.UTC))


小智 7

从新的API LocalDateTime转换为java.util.date时,以下似乎有效:

Date.from(ZonedDateTime.of({time as LocalDateTime}, ZoneId.systemDefault()).toInstant());
Run Code Online (Sandbox Code Playgroud)

反向转换可以(希望)以类似的方式实现......

希望能帮助到你...


hum*_*zed 6

如果您使用的是 android 并使用Threetenbp,则可以DateTimeUtils改用。

前任:

Date date = DateTimeUtils.toDate(localDateTime.atZone(ZoneId.systemDefault()).toInstant());
Run Code Online (Sandbox Code Playgroud)

您无法使用,Date.from因为它仅在 api 26+ 上受支持


小智 5

一切都在这里:http : //blog.progs.be/542/date-to-java-time

“往返”的答案并不准确:当你这样做时

LocalDateTime ldt = LocalDateTime.ofInstant(instant, ZoneOffset.UTC);
Run Code Online (Sandbox Code Playgroud)

如果您的系统时区不是 UTC/GMT,请更改时间!