将java.util.Date对象转换为新的JDK 8/JSR-310 java.time.LocalDate的最佳方法是什么?

Date input = new Date();
LocalDate date = ???

当前回答

简短的回答

Date input = new Date();
LocalDate date = input.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();

解释

尽管名字如此,但date表示时间线上的一个瞬间,而不是一个“日期”。对象中存储的实际数据是1970-01-01 t00:00 . 00z (1970 GMT/UTC开始的午夜时分)以来以毫秒为单位的长计数。

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

Date input = new Date();
Instant instant = input.toInstant();

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

“即时”也不包含任何时区信息。因此,要将即时日期转换为本地日期,就必须指定时区。这可能是默认区域——ZoneId.systemDefault()——也可能是应用程序控制的时区,比如用户首选项中的时区。使用atZone()方法来应用时区:

Date input = new Date();
Instant instant = input.toInstant();
ZonedDateTime zdt = instant.atZone(ZoneId.systemDefault());

zoneeddatetime包含由本地日期和时间、时区以及与GMT/UTC的偏移量组成的状态。因此,日期- LocalDate -可以使用tollocaldate()轻松提取:

Date input = new Date();
Instant instant = input.toInstant();
ZonedDateTime zdt = instant.atZone(ZoneId.systemDefault());
LocalDate date = zdt.toLocalDate();

Java 9答案

在Java SE 9中,添加了一个新方法,略微简化了这个任务:

Date input = new Date();
LocalDate date = LocalDate.ofInstant(input.toInstant(), ZoneId.systemDefault());

这个新的替代方案更直接,产生的垃圾更少,因此性能应该更好。

其他回答

如果您正在使用ThreeTen Backport,包括ThreeTenABP

    Date input = new Date(); // Imagine your Date here
    LocalDate date = DateTimeUtils.toInstant(input)
            .atZone(ZoneId.systemDefault())
            .toLocalDate();

如果您正在使用JSR 310的后端端口,要么您没有Date.toInstant()方法,要么它不会为您提供进一步转换所需的org.threeten.bp.Instant。相反,您需要使用作为后端口一部分的DateTimeUtils类。转换的剩余部分是一样的,所以解释也是一样的。

简短的回答

Date input = new Date();
LocalDate date = input.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();

解释

尽管名字如此,但date表示时间线上的一个瞬间,而不是一个“日期”。对象中存储的实际数据是1970-01-01 t00:00 . 00z (1970 GMT/UTC开始的午夜时分)以来以毫秒为单位的长计数。

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

Date input = new Date();
Instant instant = input.toInstant();

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

“即时”也不包含任何时区信息。因此,要将即时日期转换为本地日期,就必须指定时区。这可能是默认区域——ZoneId.systemDefault()——也可能是应用程序控制的时区,比如用户首选项中的时区。使用atZone()方法来应用时区:

Date input = new Date();
Instant instant = input.toInstant();
ZonedDateTime zdt = instant.atZone(ZoneId.systemDefault());

zoneeddatetime包含由本地日期和时间、时区以及与GMT/UTC的偏移量组成的状态。因此,日期- LocalDate -可以使用tollocaldate()轻松提取:

Date input = new Date();
Instant instant = input.toInstant();
ZonedDateTime zdt = instant.atZone(ZoneId.systemDefault());
LocalDate date = zdt.toLocalDate();

Java 9答案

在Java SE 9中,添加了一个新方法,略微简化了这个任务:

Date input = new Date();
LocalDate date = LocalDate.ofInstant(input.toInstant(), ZoneId.systemDefault());

这个新的替代方案更直接,产生的垃圾更少,因此性能应该更好。

如果你正在使用Java 8, @JodaStephen的答案显然是最好的。然而,如果你使用的是JSR-310后端口,你不得不做这样的事情:

Date input = new Date();
Calendar cal = Calendar.getInstance();
cal.setTime(input);
LocalDate date = LocalDate.of(cal.get(Calendar.YEAR),
        cal.get(Calendar.MONTH) + 1,
        cal.get(Calendar.DAY_OF_MONTH));

你可以在一行中转换:

public static LocalDate getLocalDateFromDate(Date date){
   return LocalDate.from(Instant.ofEpochMilli(date.getTime()).atZone(ZoneId.systemDefault()));
}

这条简单的线有什么问题?

new LocalDateTime(new Date().getTime()).toLocalDate();