2014-05-22 4 views
42

Sono un po 'scoraggiato con le date di analisi in Java 8 Time API.API Java 8 Time: come analizzare la stringa di formato "MM.yyyy" su LocalDate

In precedenza ho potuto facilmente scrivere:

String date = "04.2013"; 
DateFormat df = new SimpleDateFormat("MM.yyyy"); 
Date d = df.parse(date); 

Ma ora se io uso LocalDate e farlo in questo modo:

String date = "04.2013"; 
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("MM.yyyy"); 
LocalDate ld = LocalDate.parse(date, formatter); 

ricevo un'eccezione:

java.time.format.DateTimeParseException: Text '04' could not be parsed at index 0 
java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1948) 
java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1850) 
java.time.LocalDate.parse(LocalDate.java:400) 
java.time.LocalDate.parse(LocalDate.java:385) 
com.luxoft.ath.controllers.JsonController.region(JsonController.java:38) 
sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) 
sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
java.lang.reflect.Method.invoke(Method.java:483) 
org.springframework.web.method.support.InvocableHandlerMethod.invoke(InvocableHandlerMethod.java:215) 
org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:132) 
org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:104) 
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandleMethod(RequestMappingHandlerAdapter.java:749) 
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:689) 
org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:83) 
org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:938) 
org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:870) 
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:961) 
org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:852) 
javax.servlet.http.HttpServlet.service(HttpServlet.java:617) 
org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:837) 
javax.servlet.http.HttpServlet.service(HttpServlet.java:723) 
org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:88) 
org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107) 

Se cambio formato stringa su "aaaa-MM-gg" tutto funziona perfettamente, anche senza formattatore:

String date = "2013-04-12"; 
LocalDate ld = LocalDate.parse(date); 

Quindi la mia domanda è: come analizzare data in formato personalizzato utilizzando Java 8 Tempo API?

risposta

46

Ha senso: l'input non è realmente una data perché non dispone di informazioni sul giorno. Dovresti analizzarlo come YearMonth e usare quel risultato se non ti interessa il giorno.

String date = "04.2013"; 
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("MM.yyyy"); 
YearMonth ym = YearMonth.parse(date, formatter); 

Se si ha bisogno di applicare un giorno specifico, è possibile ottenere un LocalDate da un YearMonth ad esempio:

LocalDate ld = ym.atDay(1); 
//or 
LocalDate ld = ym.atEndOfMonth(); 

È possibile anche utilizzare un TemporalAdjuster, ad esempio, per l'ultimo giorno di mese *:

LocalDate ld = ym.atDay(1).with(lastDayOfMonth()); 

* con una import static java.time.temporal.TemporalAdjusters.lastDayOfMonth;

+2

questo ha un senso. Se rappresenti solo un anno e un mese (la data esatta è irrilevante), allora devi solo lavorare con 'YearMonth'. Non è necessario convertirlo in 'LocalDate' a differenza di pre-Java8 in cui dovevamo convertirlo in una data * predefinita * di' 1'. ':' – ADTC

+0

@ADTC In effetti ho modificato per chiarire. – assylias

+4

Si può fare 'LocalDate ld = ym.atEndOfMonth();' too .. Seriamente, Java 8 Time API (JodaTime) è sorprendente ':' – ADTC

12

seguente alternativa non è probabilmente così bello, ma almeno una soluzione sperimentata con successo, anche, in modo che ne parla qui per completezza e come supplemento alla risposta giusta di @assylias:

DateTimeFormatterBuilder builder = new DateTimeFormatterBuilder(); 
builder.parseDefaulting(ChronoField.DAY_OF_MONTH, 1); 
builder.append(DateTimeFormatter.ofPattern("MM.yyyy")); 
DateTimeFormatter dtf = builder.toFormatter(); 

String ym = "04.2013"; 
LocalDate date = LocalDate.parse(ym, dtf); 
System.out.println(date); // output: 2013-04-01