将java.util.Date转换为java.time.LocalDate

Jod*_*hen 463 java datetime java-8 java-time

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

Date input = new Date();
LocalDate date = ???
Run Code Online (Sandbox Code Playgroud)

Jod*_*hen 795

简短的回答

Date input = new Date();
LocalDate date = input.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();
Run Code Online (Sandbox Code Playgroud)

说明

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

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

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

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

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

Date input = new Date();
Instant instant = input.toInstant();
ZonedDateTime zdt = instant.atZone(ZoneId.systemDefault());
Run Code Online (Sandbox Code Playgroud)

A ZonedDateTime包含由本地日期和时间,时区以及GMT/UTC偏移量组成的状态.因此,日期 - LocalDate- 可以使用toLocalDate()以下方式轻松提取:

Date input = new Date();
Instant instant = input.toInstant();
ZonedDateTime zdt = instant.atZone(ZoneId.systemDefault());
LocalDate date = zdt.toLocalDate();
Run Code Online (Sandbox Code Playgroud)

Java 9回答

在Java SE 9中,添加了一种新方法,可以略微简化此任务:

Date input = new Date();
LocalDate date = LocalDate.ofInstant(input.toInstant(), ZoneId.systemDefault());
Run Code Online (Sandbox Code Playgroud)

这种新的替代方案更直接,产生更少的垃圾,因此应该表现更好.

  • @JodaStephen`Date`没有时区概念,`Instant`也不包含有关时区的信息.`LocalDate` API表示"没有时区的日期".那么为什么从`Date`转换为`Instant`到`LocalDate`需要`atZone(ZoneId.systemDefault())`? (22认同)
  • 我有`LocalDate.from(Instant.ofEpochMilli(date.getTime()))`我认为它等同于你的,但更直接. (13认同)
  • @assylias只需使用sqlDate.toLocalDate()! (11认同)
  • 编译但不运行的@MarkoTopolnik.Instant不包含时区,因此无法获取LocalDate. (9认同)
  • @Gustavo:`LocalDate`和`LocalDateTime`不"存储或表示时间或时区"(参考:javadocs).虽然他们不存储它 - 类确实代表*`Local`*日期和/或时间,因此转换为*local*date/time意味着时区. (6认同)
  • 为了确认,我还针对Java 8运行它并在那里失败.显然,ThreeTen实现与Java 8不同. (3认同)
  • @avalancha好吧,他在`java.time`方面确实有一点优势,是JodaTime的主要作者,也是JSR 310专家组的成员:-)他比你更深入地思考这个问题领域我可能需要! (3认同)
  • @JodaStephen java.sql.Date中使用的方法怎么样:`return LocalDate.of(getYear()+ 1900,getMonth()+ 1,getDate())`? (2认同)
  • 这个家伙产生的那种质量(帖子)令人难以置信!如果可以,我会+10 (2认同)
  • @Gustavo什么日期是新的`Date(0)`?这取决于时区.在夏威夷它的'1969-12-31`但在澳大利亚它的'1970-01-01`.这就是为什么你需要定义一个时区来进行有意义的转换.因此`ZoneId.systemDefault()`可能不正确,因为那是系统(JVM)所具有的时区. (2认同)

Oli*_*liv 148

更好的方法是:

Date date = ...;
Instant.ofEpochMilli(date.getTime()).atZone(ZoneId.systemDefault()).toLocalDate()
Run Code Online (Sandbox Code Playgroud)

这个版本的优点:

  • 无论输入是一个实例java.util.Date还是它的子类java.sql.Date(不像@ JodaStephen的方式)都可以工作.这在JDBC源数据中很常见.java.sql.Date.toInstant()总是抛出异常.

  • 对于带有JSR-310 backport的JDK8和JDK7,它也是一样的

我个人使用实用程序类(但这不是与backport兼容):

/**
 * Utilities for conversion between the old and new JDK date types 
 * (between {@code java.util.Date} and {@code java.time.*}).
 * 
 * <p>
 * All methods are null-safe.
 */
public class DateConvertUtils {

    /**
     * Calls {@link #asLocalDate(Date, ZoneId)} with the system default time zone.
     */
    public static LocalDate asLocalDate(java.util.Date date) {
        return asLocalDate(date, ZoneId.systemDefault());
    }

    /**
     * Creates {@link LocalDate} from {@code java.util.Date} or it's subclasses. Null-safe.
     */
    public static LocalDate asLocalDate(java.util.Date date, ZoneId zone) {
        if (date == null)
            return null;

        if (date instanceof java.sql.Date)
            return ((java.sql.Date) date).toLocalDate();
        else
            return Instant.ofEpochMilli(date.getTime()).atZone(zone).toLocalDate();
    }

    /**
     * Calls {@link #asLocalDateTime(Date, ZoneId)} with the system default time zone.
     */
    public static LocalDateTime asLocalDateTime(java.util.Date date) {
        return asLocalDateTime(date, ZoneId.systemDefault());
    }

    /**
     * Creates {@link LocalDateTime} from {@code java.util.Date} or it's subclasses. Null-safe.
     */
    public static LocalDateTime asLocalDateTime(java.util.Date date, ZoneId zone) {
        if (date == null)
            return null;

        if (date instanceof java.sql.Timestamp)
            return ((java.sql.Timestamp) date).toLocalDateTime();
        else
            return Instant.ofEpochMilli(date.getTime()).atZone(zone).toLocalDateTime();
    }

    /**
     * Calls {@link #asUtilDate(Object, ZoneId)} with the system default time zone.
     */
    public static java.util.Date asUtilDate(Object date) {
        return asUtilDate(date, ZoneId.systemDefault());
    }

    /**
     * Creates a {@link java.util.Date} from various date objects. Is null-safe. Currently supports:<ul>
     * <li>{@link java.util.Date}
     * <li>{@link java.sql.Date}
     * <li>{@link java.sql.Timestamp}
     * <li>{@link java.time.LocalDate}
     * <li>{@link java.time.LocalDateTime}
     * <li>{@link java.time.ZonedDateTime}
     * <li>{@link java.time.Instant}
     * </ul>
     * 
     * @param zone Time zone, used only if the input object is LocalDate or LocalDateTime.
     * 
     * @return {@link java.util.Date} (exactly this class, not a subclass, such as java.sql.Date)
     */
    public static java.util.Date asUtilDate(Object date, ZoneId zone) {
        if (date == null)
            return null;

        if (date instanceof java.sql.Date || date instanceof java.sql.Timestamp)
            return new java.util.Date(((java.util.Date) date).getTime());
        if (date instanceof java.util.Date)
            return (java.util.Date) date;
        if (date instanceof LocalDate)
            return java.util.Date.from(((LocalDate) date).atStartOfDay(zone).toInstant());
        if (date instanceof LocalDateTime)
            return java.util.Date.from(((LocalDateTime) date).atZone(zone).toInstant());
        if (date instanceof ZonedDateTime)
            return java.util.Date.from(((ZonedDateTime) date).toInstant());
        if (date instanceof Instant)
            return java.util.Date.from((Instant) date);

        throw new UnsupportedOperationException("Don't know hot to convert " + date.getClass().getName() + " to java.util.Date");
    }

    /**
     * Creates an {@link Instant} from {@code java.util.Date} or it's subclasses. Null-safe.
     */
    public static Instant asInstant(Date date) {
        if (date == null)
            return null;
        else
            return Instant.ofEpochMilli(date.getTime());
    }

    /**
     * Calls {@link #asZonedDateTime(Date, ZoneId)} with the system default time zone.
     */
    public static ZonedDateTime asZonedDateTime(Date date) {
        return asZonedDateTime(date, ZoneId.systemDefault());
    }

    /**
     * Creates {@link ZonedDateTime} from {@code java.util.Date} or it's subclasses. Null-safe.
     */
    public static ZonedDateTime asZonedDateTime(Date date, ZoneId zone) {
        if (date == null)
            return null;
        else
            return asInstant(date).atZone(zone);
    }

}
Run Code Online (Sandbox Code Playgroud)

asLocalDate()这里的方法是null安全的,toLocalDate()如果是输入则使用java.sql.Date(它可以被JDBC驱动程序覆盖以避免时区问题或不必要的计算),否则使用上述方法.

  • 如果这是更好的方式,它是非常丑陋和冗长的.如此痛苦. (10认同)
  • 与接受的答案相比,它更好,我解释了原因.是的,它很难看,但这就是为什么写了`DateConvertUtils`. (2认同)
  • 是否有Kotlin库将这些软件包打包为扩展函数,以便开发人员可以执行以下操作。日期x = ...; x.asLocalDate(); (2认同)

cek*_*ock 20

LocalDate localDate = LocalDate.parse( new SimpleDateFormat("yyyy-MM-dd").format(date) );
Run Code Online (Sandbox Code Playgroud)

  • 对局部变量的任何操作都是线程安全的. (9认同)
  • 你能告诉我为什么它不是线程安全的吗? (8认同)
  • 这里,`SimpleDateFormat`实例被*限制在当前线程中.它以线程安全的方式使用*.现在,`SimpleDateFormat`被称为'实例化昂贵'(由于它需要的所有内部数据结构),但你不能**作为'singleton'共享**(没有同步访问它),因为*它*确实不是线程安全的.(一个`ThreadLocal`解决方案可以工作*如果*代码'污染'这里的'Thread`负责线程的生命周期......但这很少发生).尴尬.避免使用`SimpleDateFormat`是*使用`javax.time`的原因. (6认同)
  • 这种方法有很多开销:'昂贵的'`SimpleDateFormat`(丢弃),中间字符串(丢弃)和解析成本.这是***解决方案,但不推荐. (6认同)
  • @ceklock SimpleDateFormat一次只能处理一个日期,如果你同时使用它,它会产生错误的结果.所以是的,这很重要.不要在全局变量中创建SimpleDateFormat的单个实例. (4认同)
  • @ceklock然后你得到的问题是它不再是线程安全的 (2认同)

dha*_*im2 19

如果您使用的是Java 8,@ JodaStephen的答案显然是最好的.但是,如果您正在使用JSR-310 backport,那么很遗憾,您必须执行以下操作:

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));
Run Code Online (Sandbox Code Playgroud)

  • 事实并非如此,@ JodaStephen的答案仍然有效.您只需要另一种方法将java.util.Date转换为Instant.为此,您可以使用org.threeten.bp.DateTimeUtils.toInstant:http://www.threeten.org/threetenbp/apidocs/org/threeten/bp/DateTimeUtils.html (4认同)
  • 当我使用backport时,DateTimeUtils不可用,但是对于使用ThreeTen Backport 1.0或更高版本的任何人都可以使用它是正确的.谢谢你指出来. (3认同)

Gus*_*avo 14

LocalDate ld = new java.sql.Date( new java.util.Date().getTime() ).toLocalDate();
Run Code Online (Sandbox Code Playgroud)


Sah*_*bra 8

您可以在一行中转换:

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


She*_*Wei 7

首先,很容易将日期转换为即时

Instant timestamp = new Date().toInstant(); 
Run Code Online (Sandbox Code Playgroud)

然后,您可以使用ofInstant()方法将Instant转换为jdk 8中的任何日期api:

LocalDateTime date = LocalDateTime.ofInstant(timestamp, ZoneId.systemDefault()); 
Run Code Online (Sandbox Code Playgroud)


MTB*_*MTB 6

为了简化将 LocalDate/time 转换为 Date 的过程,反之亦然,我创建了两个图表:

  1. 将 LocalDate 转换为 Date,反之亦然:

在此输入图像描述

  1. 将 LocalDateTime 转换为 Date,反之亦然:

在此输入图像描述


Aru*_*aaj 5

Date input = new Date();
LocalDateTime  conv=LocalDateTime.ofInstant(input.toInstant(), ZoneId.systemDefault());
LocalDate convDate=conv.toLocalDate();
Run Code Online (Sandbox Code Playgroud)

Date实例也包含时间和日期,而LocalDate没有。因此,您可以首先将其转换为LocalDateTime使用其方法,ofInstant()然后如果您想要它而没有时间,则将实例转换为LocalDate.