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