Java8相当于Joda的ISODateTimeFormat.dateTimeParser()?

Bob*_*har 7 java jodatime java-8

我无法弄清楚如何让Java8的DateTime解析行为像我试图替换的Joda等价物.问题是Joda ISODateTimeFormat.dateTimeParser();允许我输入YYYY,它仍然可以工作(2016年将成为2016-01-01T00:00:00.000Z").如何从Java8获得相同的行为?

代码很简单......

import java.time.OffsetDateTime;
import java.time.temporal.ChronoField;

import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormatter;
import org.joda.time.format.ISODateTimeFormat;

public class Java8OffsetDateTime {
  public static void main(String[] args) {
    String[] candidates =
        { "2016-11-21T17:54:51.841Z",
            "2016-11-21T09:54:51.841-08:00",
            "2016", // Java8 no can do?
            "2016-11", // Java8 no can do?
            "2016-11-21", // Java8 no can do?
            "2016-11-21T01", // Java8 no can do?
            "2016-11-21T01:02", // Java8 no can do?
            "2016-11-21T01:02:03" // Java8 no can do?
        };
    DateTimeFormatter JodaDateTimeFormatter = ISODateTimeFormat.dateTimeParser();
    for (String candidate : candidates) {
      System.out.println("\ncandidate:\t\"" + candidate + "\"");
      DateTime jodaDateTime = JodaDateTimeFormatter.parseDateTime(candidate);
      System.out.println("Joda:\t" + jodaDateTime);
      try {
        OffsetDateTime java8OffsetDateTime = OffsetDateTime.parse(candidate);
        System.out.println("Java8:\t" + java8OffsetDateTime);
        long jodaMillis = jodaDateTime.getMillis();
        long javaMillis = java8OffsetDateTime.toInstant().toEpochMilli();
        System.out.printf("jodaMillis:%d %s javaMillis:%d\n",
            jodaMillis,
            (jodaMillis==javaMillis) ? "==" : "!=",
            javaMillis);
      } catch (Exception e) {
        e.printStackTrace();
      }
    }
  }
}
Run Code Online (Sandbox Code Playgroud)

结果是挑战......

$ java -cp ~/work/joda-time-2.9.6/joda-time-2.9.6.jar:. Java8OffsetDateTime

candidate:  "2016-11-21T17:54:51.841Z"
Joda:   2016-11-21T09:54:51.841-08:00
Java8:  2016-11-21T17:54:51.841Z
jodaMillis:1479750891841 == javaMillis:1479750891841

candidate:  "2016-11-21T09:54:51.841-08:00"
Joda:   2016-11-21T09:54:51.841-08:00
Java8:  2016-11-21T09:54:51.841-08:00
jodaMillis:1479750891841 == javaMillis:1479750891841

candidate:  "2016"
Joda:   2016-01-01T00:00:00.000-08:00
java.time.format.DateTimeParseException: Text '2016' could not be parsed at index 4
    at java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1947)
    at java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1849)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:402)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:387)
    at Java8OffsetDateTime.main(Java8OffsetDateTime.java:26)

candidate:  "2016-11"
Joda:   2016-11-01T00:00:00.000-07:00
java.time.format.DateTimeParseException: Text '2016-11' could not be parsed at index 7
    at java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1947)
    at java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1849)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:402)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:387)
    at Java8OffsetDateTime.main(Java8OffsetDateTime.java:26)

candidate:  "2016-11-21"
Joda:   2016-11-21T00:00:00.000-08:00
java.time.format.DateTimeParseException: Text '2016-11-21' could not be parsed at index 10
    at java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1947)
    at java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1849)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:402)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:387)
    at Java8OffsetDateTime.main(Java8OffsetDateTime.java:26)

candidate:  "2016-11-21T01"
Joda:   2016-11-21T01:00:00.000-08:00
java.time.format.DateTimeParseException: Text '2016-11-21T01' could not be parsed at index 13
    at java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1947)
    at java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1849)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:402)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:387)
    at Java8OffsetDateTime.main(Java8OffsetDateTime.java:26)

candidate:  "2016-11-21T01:02"
Joda:   2016-11-21T01:02:00.000-08:00
java.time.format.DateTimeParseException: Text '2016-11-21T01:02' could not be parsed at index 16
    at java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1947)
    at java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1849)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:402)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:387)
    at Java8OffsetDateTime.main(Java8OffsetDateTime.java:26)

candidate:  "2016-11-21T01:02:03"
Joda:   2016-11-21T01:02:03.000-08:00
java.time.format.DateTimeParseException: Text '2016-11-21T01:02:03' could not be parsed at index 19
    at java.time.format.DateTimeFormatter.parseResolved0(DateTimeFormatter.java:1947)
    at java.time.format.DateTimeFormatter.parse(DateTimeFormatter.java:1849)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:402)
    at java.time.OffsetDateTime.parse(OffsetDateTime.java:387)
    at Java8OffsetDateTime.main(Java8OffsetDateTime.java:26)
Run Code Online (Sandbox Code Playgroud)

我怎样摆脱那些java.time.format.DateTimeParseException"Java8无法做到的?" 案件?

use*_*814 4

感谢@Tunaki 建议替代语法。这将帮助您解析您提供的输入值。相应地进行调整。

OffsetDateTime java8OffsetDateTime = OffsetDateTime.parse(candidate, offsetDateTimeFormatter );

DateTimeFormatter customOffsetDateTimeFormatter = new DateTimeFormatterBuilder().appendPattern("yyyy[-MM][-dd['T'HH[:mm[:ss]]]][.SSSXXX]")
            .parseDefaulting(ChronoField.MONTH_OF_YEAR, 1)
            .parseDefaulting(ChronoField.DAY_OF_MONTH, 1)
            .parseDefaulting(ChronoField.HOUR_OF_DAY, 0)
            .parseDefaulting(ChronoField.MINUTE_OF_HOUR, 0)
            .parseDefaulting(ChronoField.SECOND_OF_MINUTE, 0)
            .parseDefaulting(ChronoField.NANO_OF_SECOND, 0)
            .parseDefaulting(ChronoField.OFFSET_SECONDS, ZoneOffset.of("-08:00").getTotalSeconds())
            .toFormatter();
Run Code Online (Sandbox Code Playgroud)

测试用例:(-06:00 偏移)

import java.time.OffsetDateTime;
import java.time.format.DateTimeFormatter;
import java.time.format.DateTimeFormatterBuilder;
import java.time.temporal.ChronoField;
public class App 
 {
    public static void main(String[] args) {
        String[] candidates =
            {"2016-11-21T15:54:51.841Z",
                 "2016-11-21T09:54:51.841-06:00",
                    "2016", // Java8 no can do?
                    "2016-11", // Java8 no can do?
                    "2016-11-21", // Java8 no can do?
                    "2016-11-21T01", // Java8 no can do?
                    "2016-11-21T01:02", // Java8 no can do?
                    "2016-11-21T01:02:03" // Java8 no can do?*/
            };

        DateTimeFormatter customOffsetDateTimeFormatter = new DateTimeFormatterBuilder().appendPattern("yyyy[-MM][-dd['T'HH[:mm[:ss]]]][.SSSXXX]")
            .parseDefaulting(ChronoField.MONTH_OF_YEAR, 1)
            .parseDefaulting(ChronoField.DAY_OF_MONTH, 1)
            .parseDefaulting(ChronoField.HOUR_OF_DAY, 0)
            .parseDefaulting(ChronoField.MINUTE_OF_HOUR, 0)
            .parseDefaulting(ChronoField.SECOND_OF_MINUTE, 0)
            .parseDefaulting(ChronoField.NANO_OF_SECOND, 0)
            .parseDefaulting(ChronoField.OFFSET_SECONDS, ZoneOffset.of("-06:00").getTotalSeconds())
            .toFormatter();

    org.joda.time.format.DateTimeFormatter jodaDateTimeFormatter = ISODateTimeFormat.dateTimeParser();
    for (String candidate : candidates) {
        System.out.println("\ncandidate:\t\"" + candidate + "\"");
        DateTime jodaDateTime = jodaDateTimeFormatter.parseDateTime(candidate);
        System.out.println("Joda:\t" + jodaDateTime);
        try {
            OffsetDateTime java8OffsetDateTime = OffsetDateTime.parse(candidate,customOffsetDateTimeFormatter);
            System.out.println("Java8:\t" + java8OffsetDateTime);
            long jodaMillis = jodaDateTime.getMillis();
            long javaMillis = java8OffsetDateTime.toInstant().toEpochMilli();
            System.out.printf("jodaMillis:%d %s javaMillis:%d\n",
                    jodaMillis,
                    (jodaMillis == javaMillis) ? "==" : "!=",
                    javaMillis);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}  
Run Code Online (Sandbox Code Playgroud)

输出:(-06:00 偏移)

    candidate:  "2016-11-21T15:54:51.841Z"
    Joda:   2016-11-21T09:54:51.841-06:00
    Java8:  2016-11-21T15:54:51.841Z
    jodaMillis:1479743691841 == javaMillis:1479743691841

    candidate:  "2016-11-21T09:54:51.841-06:00"
    Joda:   2016-11-21T09:54:51.841-06:00
    Java8:  2016-11-21T09:54:51.841-06:00
    jodaMillis:1479743691841 == javaMillis:1479743691841

    candidate:  "2016"
    Joda:   2016-01-01T00:00:00.000-06:00
    Java8:  2016-01-01T00:00-06:00
    jodaMillis:1451628000000 == javaMillis:1451628000000

    candidate:  "2016-11"
    Joda:   2016-11-01T00:00:00.000-05:00
    Java8:  2016-11-01T00:00-06:00
    jodaMillis:1477976400000 != javaMillis:1477980000000

    candidate:  "2016-11-21"
    Joda:   2016-11-21T00:00:00.000-06:00
    Java8:  2016-11-21T00:00-06:00
    jodaMillis:1479708000000 == javaMillis:1479708000000

    candidate:  "2016-11-21T01"
    Joda:   2016-11-21T01:00:00.000-06:00
    Java8:  2016-11-21T01:00-06:00
    jodaMillis:1479711600000 == javaMillis:1479711600000

    candidate:  "2016-11-21T01:02"
    Joda:   2016-11-21T01:02:00.000-06:00
    Java8:  2016-11-21T01:02-06:00
    jodaMillis:1479711720000 == javaMillis:1479711720000

    candidate:  "2016-11-21T01:02:03"
    Joda:   2016-11-21T01:02:03.000-06:00
    Java8:  2016-11-21T01:02:03-06:00
    jodaMillis:1479711723000 == javaMillis:1479711723000
Run Code Online (Sandbox Code Playgroud)

  • 对于可选部分使用“[...]”语法会更清晰。它看起来像 `yyyy-MM-dd['T'HH[:mm[:ss]]][.SSSVV]` (2认同)
  • 我按照你提到的方式跑了。它在这里工作得很好。将测试用例添加回答案中。 (2认同)