无法使用Java 8中的DateTimeFormatter和ZonedDateTime从TemporalAccessor获取ZonedDateTime

时间:2014-05-11 18:45:27

标签: java timezone java-8 datetime-format java-time

我最近转向Java 8,希望能更轻松地处理本地和分区时间。

但是,在我看来,在解析一个简单的约会时,我面临一个简单的问题。

public static ZonedDateTime convertirAFecha(String fecha) throws Exception {
    DateTimeFormatter formatter = DateTimeFormatter.ofPattern(
            ConstantesFechas.FORMATO_DIA).withZone(
            obtenerZonaHorariaServidor());

    ZonedDateTime resultado = ZonedDateTime.parse(fecha, formatter);
    return resultado;
}

就我而言:

  • fecha是' 15/06 / 2014'
  • ConstantesFechas.FORMATO_DIA是&#dd / MM / yyyy'
  • obtenerZonaHorariaServidor返回ZoneId.systemDefault()

所以,这是一个简单的例子。但是,解析会抛出此异常:

java.time.format.DateTimeParseException: Text '15/06/2014' could not be parsed: Unable to obtain ZonedDateTime from TemporalAccessor: {},ISO resolved to 2014-06-15 of type java.time.format.Parsed

任何提示?我一直在尝试不同的解析和使用TemporalAccesor的组合,但到目前为止没有任何运气。

祝你好运

5 个答案:

答案 0 :(得分:28)

我不确定为什么它不起作用(可能是因为您的输入没有时间/时区信息)。一种简单的方法是首先将您的日期解析为LocalDate(没有时区或时区信息),然后创建ZonedDateTime

public static ZonedDateTime convertirAFecha(String fecha) {
  DateTimeFormatter formatter = DateTimeFormatter.ofPattern("dd/MM/yyyy");
  LocalDate date = LocalDate.parse(fecha, formatter);

  ZonedDateTime resultado = date.atStartOfDay(ZoneId.systemDefault());
  return resultado;
}

答案 1 :(得分:15)

这是一个错误,请参阅JDK-bug-log。根据该信息,Java 9和Java 8u20解决了这个问题。尝试下载最新的Java 8版本。今天是2014-05-12:有early access release 8u20可用。

更新:

我个人认为,因为你只有并且期望“dd / MM / yyyy”作为模式,你应该使用LocalDate作为主要类型,因为@assylias已经提出了。关于您的上下文,几乎可以肯定设计无法使用ZonedDateTime。你想用这种类型的对象做什么?我只能将专门的时区计算视为用例。而且你甚至不能直接在数据库中存储这些ZonedDateTime - 对象,所以这种类型远没有许多人所相信的那么有用。

我所描述的用例问题确实是Java-8引入的一个新方面,而不是旧的GregorianCalendar - 类(它是一体化类型)。用户必须开始考虑为他们的问题和用例选择合适的时间类型。

答案 2 :(得分:5)

简单来说就是行

ZonedDateTime.parse('2014-04-23', DateTimeFormatter.ISO_OFFSET_DATE_TIME)

抛出异常:

Text '2014-04-23' could not be parsed at index 10
java.time.format.DateTimeParseException: Text '2014-04-23' could not be parsed at index 10

对我来说这看起来像个错误。

我使用了这种解决方法:

String dateAsStr = '2014-04-23';
if (dateAsStr.length() == 10) {
    dateAsStr += 'T00:00:00';
}
ZonedDateTime.parse(dateAsStr, DateTimeFormatter.ISO_OFFSET_DATE_TIME.withZone(ZoneId.systemDefault()));

答案 3 :(得分:2)

只是一个转换示例,我相信有些人会得到以下例外

(java.time.DateTimeException: Unable to obtain LocalDateTime from TemporalAccessor: 2014-10-24T18:22:09.800Z of type java.time.Instant)

如果他们尝试

LocalDateTime localDateTime = LocalDateTime.from(new Date().toInstant());

要解决此问题,请通过区域 -

LocalDateTime localDateTime = LocalDateTime.from(new Date()
        .toInstant().atZone(ZoneId.of("UTC")));

答案 4 :(得分:2)

如果来自Google:

而不是:

ZonedDateTime.from(new Date().toInstant());

试试这个:

ZonedDateTime.ofInstant(new Date(), ZoneId.of("UTC"));