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


Answers:


828

简短答案

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

说明

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

java.util.DateJSR-310中的等效类是Instant,因此有一种方便的方法toInstant()可以提供转换:

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

一个java.util.Date实例没有时区的概念。如果调用toString(),这可能看起来很奇怪java.util.Date,因为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中,添加了一个新方法,该方法稍微简化了此任务:

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

这个新的替代方法更直接,产生更少的垃圾,因此应该表现得更好。


14
LocalDate.from(Instant.ofEpochMilli(date.getTime()))以为它等同于您的,但是更直接。
Marko Topolnik 2014年

9
@MarkoTopolnik可以编译但无法运行。Instant不包含时区,因此无法获取LocalDate。
JodaStephen 2014年

11
@assylias只需使用sqlDate.toLocalDate()即可!
JodaStephen 2014年

25
@JodaStephen Date没有时区的概念,Instant也不包含有关时区的信息。该LocalDateAPI说,“没有一个时区有个约会”。那么为什么要从Date转变InstantLocalDate需求atZone(ZoneId.systemDefault())呢?
古斯塔沃2015年

8
@Gustavo:LocalDate并且LocalDateTime不要“存储或表示时间或时区”(参考:javadocs)。尽管它们不存储-类确实代表Local日期和/或时间,因此转换为本地日期/时间意味着时区。
Cuga 2015年

158

更好的方法是:

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

此版本的优点:

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

  • 具有JSR-310反向端口的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);
    }

}

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


12
如果这是更好的方法,则它非常丑陋且冗长。如此痛苦。
ceklock 2015年

3
与接受的答案相比,它更好,我解释了原因。是的,这很丑陋,但这就是为什么写DateConvertUtils
奥利夫,2015年

我不明白为什么他们没有在新的API中实现转换类。
ceklock

@ceklock,他们确实实现了,不是实现转换类,而是实现了几种转换方法,例如Date.toInstant()
Ole VV

2
是否有Kotlin库将这些软件包打包为扩展函数,以便开发人员可以执行以下操作。日期x = ...; x.asLocalDate();
Mark Ashworth

20
LocalDate localDate = LocalDate.parse( new SimpleDateFormat("yyyy-MM-dd").format(date) );

7
在这里,SimpleDateFormat实例仅限于当前线程。它是用于在一个线程安全的方式。现在,SimpleDateFormat被称为“实例化的昂贵”(由于它需要所有内部数据结构),但是您不能一个实例共享为“单个”(不同步访问),因为确实不是线程-安全。(如果此代码中的“污染”代码负责线程的生命周期,ThreadLocal解决方案可以工作,但是这种情况很少发生)。尴尬。避免是使用原因。ThreadSimpleDateFormatjavax.time
大卫·布洛克

6
这种方法有很多开销:“昂贵” SimpleDateFormat(被丢弃),中间字符串(被丢弃)以及解析成本。这是一个解决方案,但不建议这样做。
大卫·布洛克

2
@ceklock,但随后您得到的问题是它不再是线程安全的
flup

4
@ ceklock,SimpleDateFormat一次只能处理一个日期,如果同时使用,它将产生错误的结果。是的,这很重要。不要在全局变量中创建SimpleDateFormat的单个实例。
flup

19

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

4
不正确,@ JodaStephen的答案仍然有效。您只需要将java.util.Date转换为Instant的另一种方法。:为此,您可以使用org.threeten.bp.DateTimeUtils.toInstant threeten.org/threetenbp/apidocs/org/threeten/bp/...
基督教Ciach

3
当我使用反向端口时,DateTimeUtils不可用,但是您正确地知道使用ThreeTen Backport 1.0或更高版本的任何人都可以使用它。感谢您指出。
dhalsim2

14
LocalDate ld = new java.sql.Date( new java.util.Date().getTime() ).toLocalDate();

1
最简单的实现:LocalDate.of(getYear()+ 1900,getMonth()+ 1,getDate())
Grigory Kislin

虽然说不赞成的行:|
TheRealChx101

8

您可以在一行中进行转换:

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

我使用这种方法收到这样的错误:java.time.DateTimeException:无法从TemporalAccessor获取LocalDate:2018-01-31T11:54:27.964Z,类型为java.time.Instant
Luigi Rubino,

@LuigiRubino感谢您指出。请查看更新后的答案。我忘记了较早添加区域。
Sahil Chhabra

8

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

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

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

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

包括localDate,localDateTime,localTime
Shedom Wei

在此处指定ZoneId是什么意思。如果我从API获取Date-> Instant,我可以将其视为UTC的1970年以来的毫秒数。当我想从api的角度仅获取对应的LocalDate(yyyy-mm-dd)而不转换为任何时区时,我不应该使用ZoneOffset.UTC将此即时偏移量偏移到我的本地时区。您的ZoneId.systemDefault()示例不是按日期转换服务器。例如 Instant代表2018-10-20 0:00,然后从UTC转到America / Los_Angeles我到达的是本地日期2018-10-19,而不是2018-10-20?
米哈尔Ziobro

如果碰巧import java.sql.Date在文件中,它将中断:始终抛出的toInstant()方法java.sql.Date
奥利夫,

4
Date input = new Date();
LocalDateTime  conv=LocalDateTime.ofInstant(input.toInstant(), ZoneId.systemDefault());
LocalDate convDate=conv.toLocalDate();

Date实例确实包含时间以及日期,而LocalDate没有。因此,您可以首先LocalDateTime使用它的方法将其转换为,ofInstant()然后,如果您想在没有时间的情况下将其转换为LocalDate


1
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();
    }

0

我在JBoss EAP 6上使用@JodaStephen实现时遇到了问题。因此,我按照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();

-2

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

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

Java抱怨说它不能在基本类型上长时间调用toLocalDate()。我使用了实际的Date对象;也许有摩擦??
TheGeeko61'3

-8

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

  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”格式打印日期


1
问题是使用java.timeJDK8中的类而不是Joda Time 寻找解决方案。
pioto
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.