Come posso convertire in modo ottimale un java.util.Date
in un Java 8 java.time.YearMonth
?Come convertire java.util.Date in Java8 java.time.YearMonth
Purtroppo il seguente tiri un DateTimeException
:
YearMonth yearMonth = YearMonth.from(date.toInstant());
risultati in:
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)
...
ho bisogno di questa funzionalità perché voglio conservare YearMonth
valori in un database utilizzando JPA. Attualmente JPA non supporta YearMonth
's, così mi è venuta in mente la seguente YearMonthConverter (importazioni omessi):
// 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);
}
}
Non c'è una soluzione migliore (più pulito, più breve) (per entrambe le direzioni)?
Personalmente eviterei mappatura a 'java.util.Date 'perché a) non è' java.sql. * '-tipo, b) include il fuso orario implicito (prestazioni scadenti e uso sfocato del fuso orario del sistema), c) SQL non definisce tale tipo per anno-mese. Meglio mappare 'YearMonth' su intero usando [PROLEPTIC_MONTH] (http://docs.oracle.com/javase/8/docs/api/java/time/temporal/ChronoField.html#PROLEPTIC_MONTH) –
@MenoHochschild: questa è una buona idea - solo l'avvertenza è che non è leggibile quando si guarda al database stesso. Nessun problema nel mio caso ... – Sebastian
Beh, solo un numero intero non è leggibile per la visualizzazione diretta del contenuto di db, ma attenzione, il mapping a juDate può cambiare la data e il mese visualizzati (come l'amministratore di db lo vede) a causa del fuso orario conversioni e impostazioni db/server. –