Java 8 - конвертировать дату в LocalDate и LocalDateTime

Java 8 - преобразование даты в LocalDate и LocalDateTime

java.util.Date не имеет понятия часового пояса и представляет только количество секунд, прошедших с моментаUnix epoch time - 1970-01-01T00: 00: 00Z (полночь в начале 1 января 1970 г. по Гринвичу / УНИВЕРСАЛЬНОЕ ГЛОБАЛЬНОЕ ВРЕМЯ)

Note
Новый класс Java 8java.time.Instant эквивалентен классическомуjava.util.Date

1. Дата → java.time

Идея преобразования даты:

Date -> Instant + System default time zone = LocalDate
Date -> Instant + System default time zone = LocalDateTime
Date -> Instant + System default time zone = ZonedDateTime

В этом примере показано, как преобразоватьjava.util.Date в новые API Java 8 Date -LocalDate,LocalDateTime иZonedDateTime.

DateToJavaTime.java

package com.example.java8;

import java.time.*;
import java.util.Date;

public class DateToJavaTime {

    public static void main(String[] args) {

        //Asia/Kuala_Lumpur +8
        ZoneId defaultZoneId = ZoneId.systemDefault();
        System.out.println("System Default TimeZone : " + defaultZoneId);

        //toString() append +8 automatically.
        Date date = new Date();
        System.out.println("date : " + date);

        //1. Convert Date -> Instant
        Instant instant = date.toInstant();
        System.out.println("instant : " + instant); //Zone : UTC+0

        //2. Instant + system default time zone + toLocalDate() = LocalDate
        LocalDate localDate = instant.atZone(defaultZoneId).toLocalDate();
        System.out.println("localDate : " + localDate);

        //3. Instant + system default time zone + toLocalDateTime() = LocalDateTime
        LocalDateTime localDateTime = instant.atZone(defaultZoneId).toLocalDateTime();
        System.out.println("localDateTime : " + localDateTime);

        //4. Instant + system default time zone = ZonedDateTime
        ZonedDateTime zonedDateTime = instant.atZone(defaultZoneId);
        System.out.println("zonedDateTime : " + zonedDateTime);

    }

}

Выход

System Default TimeZone : Asia/Kuala_Lumpur

date : Fri Aug 19 21:46:31 MYT 2016
instant : 2016-08-19T13:46:31.981Z

localDate : 2016-08-19
localDateTime : 2016-08-19T21:46:31.981
zonedDateTime : 2016-08-19T21:46:31.981+08:00[Asia/Kuala_Lumpur]

2. Объяснение - Q&A

2.1 Question: ЕслиDate не имеет понятия о часовом поясе, почему часовой пояс будет отображаться, когда мы распечатываем объектDate? Например :

//Fri Aug 19 11:52:06 MYT 2016
System.out.println(new Date()); //MYT = my system default time zone

Answer: проверьте исходный кодjava.uti.Date.toString(), если вы распечатаете объектDate, часовой пояс системы по умолчанию будет добавлен и отображен вместе.

java.util.Date

public String toString() {

        //...omitted...

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

Note
Такое поведение является недостатком дизайна, начиная с JDK1.1, оно вызывает большую путаницу. Опять же,java.util.Date не хранит информацию о часовом поясе, но если вы распечатаете ее, часовой пояс системы по умолчанию будет отображаться вместе.

2.2 Question: для преобразованияDate, почему нам нужно добавить системный часовой пояс по умолчанию дляjava.time.instant?
Answer: см. выше 2.1 вопросы и ответы. Просмотрите еще один пример:

1. Date = 19/08/2016T10:00:00
2. System default time zone = +08:00 [Asia/Kuala_Lumpur]
3. Date (Print) = 19/08/2016T10:00:00+08:00 = 19/08/2016T18:00:00

Цель преобразования - убедиться, что и printDate, и printLocalDate будут генерировать одинаковый результат.

// Assume 19/08/2016T10:00:00 = 1000
// System default time zone = +8

1. Date (1000) -> Print Date (1000) = 1000+08:00
// we always see "1000+08:00" (but the Date is still 1000)

2. Date (1000) -> Instant (1000)
// instant has no time zone or zero offset (UTC+0/Z)

3. Instant(1000) -> LocalDate(1000) -> Print LocalDate(1000) = 1000
// The result is "1000", different with print date!

4. LocalDate(1000) + 08:00 -> LocalDate(1000+08:00)
// add default time zone +8

5. Print LocalDate(1000+08:00) = 1000+08:00

3. java.time → Date

В этом примере показано, как преобразоватьLocalDate,LocalDateTime иZonedDateTime обратно в классическийjava.util.Date.

JavaTimeToDate.java

package com.example.java8;

import java.time.*;
import java.util.Date;

public class JavaTimeToDate {

    public static void main(String[] args) {

        //Asia/Kuala_Lumpur +8
        ZoneId defaultZoneId = ZoneId.systemDefault();
        System.out.println("System Default TimeZone : " + defaultZoneId);

        LocalDate localDate = LocalDate.of(2016, 8, 19);
        Date date = Date.from(localDate.atStartOfDay(defaultZoneId).toInstant());
        System.out.println("\n1. LocalDate -> Date");
        System.out.println("localDate : " + localDate);
        System.out.println("date : " + date);

        LocalDateTime localDateTime = LocalDateTime.of(2016,8,19,21,46,31);
        Date date2 = Date.from(localDateTime.atZone(defaultZoneId).toInstant());
        System.out.println("\n2. LocalDateTime -> Date");
        System.out.println("localDateTime : " + localDateTime);
        System.out.println("date2 : " + date2);

        ZonedDateTime zonedDateTime = localDateTime.atZone(defaultZoneId);
        Date date3 = Date.from(zonedDateTime.toInstant());
        System.out.println("\n3. ZonedDateTime -> Date");
        System.out.println("zonedDateTime : " + zonedDateTime);
        System.out.println("date3 : " + date3);

    }

}

Выход

System Default TimeZone : Asia/Kuala_Lumpur

1. LocalDate -> Date
localDate : 2016-08-19
date : Fri Aug 19 00:00:00 MYT 2016

2. LocalDateTime -> Date
localDateTime : 2016-08-19T21:46:31
date2 : Fri Aug 19 21:46:31 MYT 2016

3. ZonedDateTime -> Date
zonedDateTime : 2016-08-19T21:46:31+08:00[Asia/Kuala_Lumpur]
date3 : Fri Aug 19 21:46:31 MYT 2016