【问题标题】:Convert java.util.Date to java.time.LocalDate将 java.util.Date 转换为 java.time.LocalDate
【发布时间】:2014-02-10 02:26:09
【问题描述】:

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

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

【问题讨论】:

    标签: java datetime java-8 java-time


    【解决方案1】:

    简答

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

    说明

    尽管有它的名字,java.util.Date 代表时间线上的一个瞬间,而不是“日期”。对象中存储的实际数据是自 1970-01-01T00:00Z(格林威治标准时间/UTC 开始时的午夜)以来的 long 毫秒计数。

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

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

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

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

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

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

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

    Java 9 答案

    在 Java SE 9 中,添加了 new method 以稍微简化此任务:

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

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

    【讨论】:

    • 我有LocalDate.from(Instant.ofEpochMilli(date.getTime())) 我觉得和你的一样,但是更直接。
    • @MarkoTopolnik 编译但不运行。 Instant 不包含时区,因此无法获取 LocalDate。
    • @assylias 只需使用 sqlDate.toLocalDate() !
    • @JodaStephen Date 没有时区的概念,Instant 也不包含有关时区的信息。 LocalDate API 表示“没有时区的日期”。那为什么从Date 转换为InstantLocalDate 需要atZone(ZoneId.systemDefault())
    • @Gustavo: LocalDateLocalDateTime 不“存储或表示时间或时区”(参考:javadocs)。虽然他们不存储它 - 这些类确实代表 Local 日期和/或时间,因此转换为 local 日期/时间意味着时区。跨度>
    【解决方案2】:

    更好的方法是:

    Date date = ...;
    Instant.ofEpochMilli(date.getTime()).atZone(ZoneId.systemDefault()).toLocalDate()
    

    此版本的优点:

    • 无论输入是java.util.Date 的实例还是java.sql.Date 的子类都有效(与@JodaStephen 的方式不同)。这对于源自 JDBC 的数据很常见。 java.sql.Date.toInstant() 总是抛出异常。

    • JDK8 和 JDK7 与 JSR-310 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);
        }
    
    }
    

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

    【讨论】:

    • 如果这是更好的方法,它是非常丑陋和冗长的。好痛苦。
    • 与公认的答案相比更好,我解释了原因。是的,它很丑,但这就是为什么写DateConvertUtils
    • 我不明白他们为什么不在新 API 中实现转换类。
    • @ceklock,他们确实实现了,不是一个转换类,而是几个转换方法,比如Date.toInstant()
    • 是否有 Kotlin 库将这些打包为扩展函数,以便开发人员可以执行以下操作。日期 x = ...; x.asLocalDate();
    【解决方案3】:
    LocalDate localDate = LocalDate.parse( new SimpleDateFormat("yyyy-MM-dd").format(date) );
    

    【讨论】:

    • 这里,SimpleDateFormat 实例被限制到当前线程。它以线程安全的方式使用。现在,SimpleDateFormat 被认为是“实例化昂贵”(考虑到它需要的所有内部数据结构),但您不能共享一个作为“单例”(不同步访问it),因为 it 确实不是线程安全的。 (ThreadLocal 解决方案可以工作如果代码“污染”Thread 负责线程的生命周期......但这种情况很少发生)。尴尬的。避免使用SimpleDateFormat 使用javax.time 的原因。
    • 这种方法有很多开销:'昂贵的'SimpleDateFormat(被丢弃)、中间字符串(被丢弃)和解析成本。这是一个解决方案,但不推荐。
    • @ceklock 但是你会遇到问题,它不再是线程安全的
    • @ceklock SimpleDateFormat 一次只能处理一个日期,如果你同时使用它,它会产生混乱的结果。所以,是的,这很重要。不要在全局变量中创建 SimpleDateFormat 的单个实例。
    【解决方案4】:

    如果您使用的是 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));
    

    【讨论】:

    • 不正确,@JodaStephen 的回答仍然有效。您只需要另一种方法将 java.util.Date 转换为 Instant。为此,您可以使用 org.threeten.bp.DateTimeUtils.toInstant:threeten.org/threetenbp/apidocs/org/threeten/bp/…
    • DateTimeUtils 在我使用 backport 时不可用,但您是正确的,任何使用 ThreeTen Backport 1.0 或更高版本的人都可以使用它。感谢您指出。
    • @ChristianCiach 你是对的。我在this answer 中说明了你在说什么。
    【解决方案5】:
    LocalDate ld = new java.sql.Date( new java.util.Date().getTime() ).toLocalDate();
    

    【讨论】:

    • 最简单的实现:LocalDate.of(getYear() + 1900, getMonth() + 1, getDate())
    • 那些说不推荐使用的行:|
    【解决方案6】:

    你可以在一行中转换:

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

    【讨论】:

    • 我使用这个方法得到一个这样的错误:java.time.DateTimeException: Unable to get LocalDate from TemporalAccessor: 2018-01-31T11:54:27.964Z of type java.time.Instant跨度>
    • @LuigiRubino 感谢您的指出。请参阅更新的答案。我之前忘记添加区域了。
    【解决方案7】:

    首先,将 Date 转换为 Instant 很容易

    Instant timestamp = new Date().toInstant(); 
    

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

    LocalDateTime date = LocalDateTime.ofInstant(timestamp, ZoneId.systemDefault()); 
    

    【讨论】:

    • 包括localDate、localDateTime、localTime
    • 这里指定ZoneId是什么意思。如果我从 API 获得 Date -> Instant ,我可以将其视为 1970 年 UTC 的毫秒数。当我想从未转换为任何时区的 api 的角度获取相应的 LocalDate (yyyy-mm-dd) 时,我不应该使用 ZoneOffset.UTC 来避免我的本地时区的即时偏移。您的 ZoneId.systemDefault() 示例不是正在转换日期表单服务器吗?前任。即时代表 2018-10-20 0:00,然后从 UTC 转移到 America/Los_Angeles 我在本地日期 2018-10-19 而不是 2018-10-20?
    • 如果你的文件中碰巧有import java.sql.Date,它就会中断:java.sql.DatetoInstant() 方法总是抛出。
    【解决方案8】:
    Date input = new Date();
    LocalDateTime  conv=LocalDateTime.ofInstant(input.toInstant(), ZoneId.systemDefault());
    LocalDate convDate=conv.toLocalDate();
    

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

    【讨论】:

      【解决方案9】:
      public static LocalDate Date2LocalDate(Date date) {
              return LocalDate.parse(date.toString(), DateTimeFormatter.ofPattern("EEE MMM dd HH:mm:ss zzz yyyy"))
      

      此格式来自Date#tostring

          public String toString() {
              // "EEE MMM dd HH:mm:ss zzz yyyy";
              BaseCalendar.Date date = normalize();
              StringBuilder sb = new StringBuilder(28);
              int index = date.getDayOfWeek();
              if (index == BaseCalendar.SUNDAY) {
                  index = 8;
              }
              convertToAbbr(sb, wtb[index]).append(' ');                        // EEE
              convertToAbbr(sb, wtb[date.getMonth() - 1 + 2 + 7]).append(' ');  // MMM
              CalendarUtils.sprintf0d(sb, date.getDayOfMonth(), 2).append(' '); // dd
      
              CalendarUtils.sprintf0d(sb, date.getHours(), 2).append(':');   // HH
              CalendarUtils.sprintf0d(sb, date.getMinutes(), 2).append(':'); // mm
              CalendarUtils.sprintf0d(sb, date.getSeconds(), 2).append(' '); // ss
              TimeZone zi = date.getZone();
              if (zi != null) {
                  sb.append(zi.getDisplayName(date.isDaylightTime(), TimeZone.SHORT, Locale.US)); // zzz
              } else {
                  sb.append("GMT");
              }
              sb.append(' ').append(date.getYear());  // yyyy
              return sb.toString();
          }
      

      【讨论】:

        【解决方案10】:

        @JodaStephen 在 JBoss EAP 6 上的实现遇到了问题。所以,我按照 http://docs.oracle.com/javase/tutorial/datetime/iso/legacy.html 中的 Oracle 的 Java 教程重写了转换。

            Date input = new Date();
            GregorianCalendar gregorianCalendar = (GregorianCalendar) Calendar.getInstance();
            gregorianCalendar.setTime(input);
            ZonedDateTime zonedDateTime = gregorianCalendar.toZonedDateTime();
            zonedDateTime.toLocalDate();
        

        【讨论】:

          【解决方案11】:

          如果您使用的是 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。相反,您需要使用作为 backport 一部分的 DateTimeUtils 类。其余的转换是一样的,所以解释也是。

          【讨论】:

            【解决方案12】:

            这简单的 1 行有什么问题?

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

            【讨论】:

            • Java 抱怨它不能在原始类型 long 上调用 toLocalDate()。我使用了一个实际的 Date 对象;也许有问题??
            【解决方案13】:

            我用下面的解决方案解决了这个问题

              import org.joda.time.LocalDate;
              Date myDate = new Date();
              LocalDate localDate = LocalDate.fromDateFields(myDate);
              System.out.println("My date using Date" Nov 18 11:23:33 BRST 2016);
              System.out.println("My date using joda.time LocalTime" 2016-11-18);
            

            在这种情况下,localDate 以这种格式“yyyy-MM-dd”打印您的日期

            【讨论】:

            • 问题是在 JDK8 中寻找使用 java.time 类的解决方案,而不是使用 Joda Time。
            猜你喜欢
            • 1970-01-01
            • 1970-01-01
            • 2018-10-31
            • 1970-01-01
            • 2015-01-30
            • 2011-08-29
            • 2019-08-03
            相关资源
            最近更新 更多