2015-01-08 6 views
9

Как я могу наилучшим образом преобразовать java.util.Date в Java 8 java.time.YearMonth?Как преобразовать java.util.Date в Java8 java.time.YearMonth

К сожалению следующее бросает DateTimeException:

YearMonth yearMonth = YearMonth.from(date.toInstant()); 

приводит:

java.time.DateTimeException: Unable to obtain YearMonth from TemporalAccessor: 2015-01-08T14:28:39.183Z of type java.time.Instant 
at java.time.YearMonth.from(YearMonth.java:264) 
... 

мне нужна эта функция, так как я хочу, чтобы хранить YearMonth значений в базе данных с помощью JPA. В настоящее время JPA не поддерживает YearMonth «S, поэтому я придумал следующий YearMonthConverter (импорт опущена):

// TODO (future): delete when next version of JPA (i.e. Java 9?) supports YearMonth. See https://java.net/jira/browse/JPA_SPEC-63 
@Converter(autoApply = true) 
public class YearMonthConverter implements AttributeConverter<YearMonth, Date> { 

    @Override 
    public Date convertToDatabaseColumn(YearMonth attribute) { 
    // uses default zone since in the end only dates are needed 
    return attribute == null ? null : Date.from(attribute.atDay(1).atStartOfDay(ZoneId.systemDefault()).toInstant()); 
    } 

    @Override 
    public YearMonth convertToEntityAttribute(Date dbData) { 
    // TODO: check if Date -> YearMonth can't be done in a better way 
    if (dbData == null) return null; 
    Calendar calendar = Calendar.getInstance(); 
    calendar.setTime(dbData); 
    return YearMonth.of(calendar.get(Calendar.YEAR), calendar.get(Calendar.MONTH) + 1); 
    } 
} 

не лучший (уборщик, короче) раствор (в обоих направлениях)?

+1

Лично я хотел бы избежать отображение в 'java.util.Date 'потому что a) это не' java.sql. * '-type, b) включает неявный часовой пояс (плохая производительность и нечеткое использование системного часового пояса), c) SQL не определяет такой тип для года. Лучше сопоставить «YearMonth» с целым числом с помощью [PROLEPTIC_MONTH] (http://docs.oracle.com/javase/8/docs/api/java/time/temporal/ChronoField.html#PROLEPTIC_MONTH) –

+0

@MenoHochschild: Это хорошая идея - только оговорка заключается в том, что она не читается при просмотре самой базы данных. Нет проблем в моем случае ... – Sebastian

+0

Ну, просто целое число не читается для прямого просмотра содержимого db, но имейте в виду, что отображение в juDate может изменить визуализированную дату и месяц (как видит администратор db) из-за часового пояса конверсий и настроек db/server. –

ответ

13

Короткий ответ:

// From Date to YearMonth 
YearMonth yearMonth = 
     YearMonth.from(date.toInstant() 
          .atZone(ZoneId.systemDefault()) 
          .toLocalDate()); 

// From YearMonth to Date 
// The same as the OP:s answer 
final Date convertedFromYearMonth = 
     Date.from(yearMonth.atDay(1).atStartOfDay(ZoneId.systemDefault()).toInstant()); 

Объяснение:

JavaDoc из YearMonth.from(TemporalAccessor) -метод говорит:

Конверсия извлекает YEAR и MONTH_OF_YEAR поля. Выделение разрешено только в том случае, если временный объект имеет хронологию ISO или может быть преобразован в LocalDate.

Таким образом, вы должны либо быть в состоянии:

  1. экстракта YEAR и MONTH_OF_YEAR поля или
  2. вы должны использовать что-то, что может быть конвертировано в LocalDate.

Попробуй!

final Date date = new Date(); 
final Instant instant = date.toInstant(); 
instant.get(ChronoField.YEAR); // causes an error 

Это не представляется возможным, возникает исключение:

java.time.temporal.UnsupportedTemporalTypeException: неподдерживаемое поля: Год на java.time.Instant.get (Instant.java:571) ...

Это означает, что альтернатива 1 выходит из окна. Причина этого объясняется в этом превосходном ответе о how to convert Date to LocalDate.

Несмотря на свое название, java.util.Date представляет собой мгновение на линии времени, а не «дату». Фактические данные, хранящиеся в объекте, представляют собой длинные миллисекунды с 1970-01-01T00: 00Z (полночь в начале 1970 года по Гринвичу/UTC).

Эквивалентный класс java.util.Date в JSR-310 является мгновенным, поэтому для обеспечения преобразования используется удобный метод toInstant().

Таким образом, Date может быть преобразован в Instant но это не помогло нам, не так ли?

Альтернатива 2, однако, оказывается успешной. Преобразуйте Instant в LocalDate, а затем используйте метод YearMonth.from(TemporalAccessor).

Date date = new Date(); 

    LocalDate localDate = date.toInstant() 
           .atZone(ZoneId.systemDefault()) 
           .toLocalDate(); 

    YearMonth yearMonth = YearMonth.from(localDate); 
    System.out.println("YearMonth: " + yearMonth); 

Выходной сигнал является (так как код был выполнен в январе 2015 года;):

годМесяц: 2015-01

Смежные вопросы