【问题标题】:Format LocalDateTime with Timezone in Java8在 Java8 中使用时区格式化 LocalDateTime
【发布时间】:2014-10-23 01:23:51
【问题描述】:

我有这个简单的代码:

DateTimeFormatter FORMATTER = DateTimeFormatter.ofPattern("yyyyMMdd HH:mm:ss.SSSSSS Z");
LocalDateTime.now().format(FORMATTER)

然后我会得到以下异常:

java.time.temporal.UnsupportedTemporalTypeException: Unsupported field: OffsetSeconds
at java.time.LocalDate.get0(LocalDate.java:680)
at java.time.LocalDate.getLong(LocalDate.java:659)
at java.time.LocalDateTime.getLong(LocalDateTime.java:720)
at java.time.format.DateTimePrintContext.getValue(DateTimePrintContext.java:298)
at java.time.format.DateTimeFormatterBuilder$OffsetIdPrinterParser.format(DateTimeFormatterBuilder.java:3315)
at java.time.format.DateTimeFormatterBuilder$CompositePrinterParser.format(DateTimeFormatterBuilder.java:2182)
at java.time.format.DateTimeFormatter.formatTo(DateTimeFormatter.java:1745)
at java.time.format.DateTimeFormatter.format(DateTimeFormatter.java:1719)
at java.time.LocalDateTime.format(LocalDateTime.java:1746)

如何解决这个问题?

【问题讨论】:

    标签: java java-8 java-time


    【解决方案1】:

    LocalDateTime 是没有时区的日期时间。您在格式中指定了时区偏移格式符号,但是,LocalDateTime 没有此类信息。这就是发生错误的原因。

    如果你想要时区信息,你应该使用ZonedDateTime

    DateTimeFormatter FORMATTER = DateTimeFormatter.ofPattern("yyyyMMdd HH:mm:ss.SSSSSS Z");
    ZonedDateTime.now().format(FORMATTER);
    => "20140829 14:12:22.122000 +09"
    

    【讨论】:

    • 超级有帮助。不确定我是否会如此轻松地抓住时区。很好的答案。谢谢!
    【解决方案2】:

    JSR-310 中的前缀“Local”(在 Java-8 中也称为 java.time-package)并不表示该类的内部状态中存在时区信息(此处为:LocalDateTime)。尽管名称经常具有误导性,但像 LocalDateTimeLocalTime 这样的类没有时区信息或偏移量

    您尝试使用偏移信息(由模式符号 Z 指示)格式化此类时间类型(不包含任何偏移)。因此格式化程序尝试访问不可用的信息,并且必须抛出您观察到的异常。

    解决方案:

    使用具有此类偏移或时区信息的类型。在 JSR-310 中,这是 OffsetDateTime(包含偏移量但不包含包含 DST 规则的时区)或 ZonedDateTime。您可以通过查找方法isSupported(TemporalField). 来查看此类类型的所有受支持字段。 OffsetSeconds 字段在 OffsetDateTimeZonedDateTime 中受支持,但在 LocalDateTime 中不支持。

    DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyyMMdd HH:mm:ss.SSSSSS Z");
    String s = ZonedDateTime.now().format(formatter);
    

    【讨论】:

    • 你会用什么前缀代替前缀“Local”以避免混淆?
    • @Matthew 还有一个由瑞士信贷银行发起的debate 建议名称为PlainDateTime 等。可能更好,因为前缀“plain”确实表明除了日期时间之外什么都没有.如果我们还在 Java v1.0 之前,那么没有前缀会更好,但是像 Date 这样的名称已经被旧 JDK 保留。