将 Instant 格式化为字符串

Posted

技术标签:

【中文标题】将 Instant 格式化为字符串【英文标题】:UnsupportedTemporalTypeException when formatting Instant to String 【发布时间】:2014-10-03 10:42:20 【问题描述】:

我正在尝试使用新的 java 8 time-api 和模式:

Instant instant = ...;
String out = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss").format(instant);

使用上面的代码我得到一个异常,它抱怨一个不受支持的字段:

java.time.temporal.UnsupportedTemporalTypeException: Unsupported field: YearOfEra
    at java.time.Instant.getLong(Instant.java:608)
    at java.time.format.DateTimePrintContext.getValue(DateTimePrintContext.java:298)
    ...

【问题讨论】:

【参考方案1】:

时区

要格式化Instant,需要time-zone。如果没有时区,格式化程序不知道如何将即时转换为人工日期时间字段,因此会引发异常。

时区可以使用withZone()直接添加到格式化程序中。

DateTimeFormatter formatter =
    DateTimeFormatter.ofLocalizedDateTime( FormatStyle.SHORT )
                     .withLocale( Locale.UK )
                     .withZone( ZoneId.systemDefault() );

如果您特别想要没有明确时区的 ISO-8601 格式 (正如 OP 所要求的),时区隐含 UTC,你需要

DateTimeFormatter.ISO_LOCAL_DATE_TIME.withZone(ZoneId.from(ZoneOffset.UTC))

生成字符串

现在使用该格式化程序生成 Instant 的字符串表示形式。

Instant instant = Instant.now();
String output = formatter.format( instant );

转储到控制台。

System.out.println("formatter: " + formatter + " with zone: " + formatter.getZone() + " and Locale: " + formatter.getLocale() );
System.out.println("instant: " + instant );
System.out.println("output: " + output );

运行时。

formatter: Localized(SHORT,SHORT) with zone: US/Pacific and Locale: en_GB
instant: 2015-06-02T21:34:33.616Z
output: 02/06/15 14:34

【讨论】:

谢谢!!顺便说一句,指向例如年份的异常“不受支持的字段”非常迟钝。也许应该检测到这种情况,并且应该抛出一个直接指向 Instant 中缺少的区域 id 的异常! 更奇怪的是,如果您包含 .withZone(例如 .withZone(ZoneId.of("Z")) )并格式化 LocalDateTime,则该区域将被忽略!因此,只要包含 .withZone(),相同的格式化程序就可以用于 Instant 和 LocalDateTime,而不会影响后者显示的时间。 为什么很难接受 Instant 的时区是 GMT 的事实? @KorayTugay 因为虽然迂腐的“Instant 已经是 GMT”cmets 可能是真的,但在面临抛出异常跟踪时它们远没有帮助,因为格式化 Instant 没有指定时区不起作用。如果格式化程序默认为 GMT 会很好,但是哦。 现在给你yyyy-MM-dd hh:mm:ss 格式:DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss").withZone(ZoneId.of("Europe/Paris"));【参考方案2】:
public static void main(String[] args) 

    DateTimeFormatter DATE_TIME_FORMATTER = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")
            .withZone(ZoneId.systemDefault());

    System.out.println(DATE_TIME_FORMATTER.format(new Date().toInstant()));


【讨论】:

虽然此代码可以回答问题,但提供有关 如何为什么 解决问题的附加上下文将提高​​答案的长期价值。 虽然这段代码 sn-p 可以解决问题,但including an explanation 确实有助于提高帖子的质量。请记住,您是在为将来的读者回答问题,而这些人可能不知道您提出代码建议的原因。 new Date().toInstant() 可以替换为 Instant.now()【参考方案3】:
DateTimeFormatter.ISO_INSTANT.format(Instant.now())

这使您不必转换为 UTC。但是,其他一些语言的时间框架可能不支持毫秒,所以你应该这样做

DateTimeFormatter.ISO_INSTANT.format(Instant.now().truncatedTo(ChronoUnit.SECONDS))

【讨论】:

“其他语言的时间框架”是什么意思? ISO_INSTANT.format() 会自动截断到秒吗? 一些框架只希望时间达到秒,因为它们不遵循完整的 ISO 约定,并且在输入字符串中有毫秒时会中断。【参考方案4】:

Instant 类不包含区域信息,它仅存储来自 UNIX 纪元的时间戳(以毫秒为单位),即来自 UTC 的 1070 年 1 月 1 日。 因此,格式化程序无法打印日期,因为日期总是为具体时区打印。 您应该将时区设置为格式化程序,一切都会好起来的,如下所示:

Instant instant = Instant.ofEpochMilli(92554380000L);
DateTimeFormatter formatter = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.SHORT).withLocale(Locale.UK).withZone(ZoneOffset.UTC);
assert formatter.format(instant).equals("07/12/72 05:33");
assert instant.toString().equals("1972-12-07T05:33:00Z");

【讨论】:

纳秒,而不是毫秒【参考方案5】:

时间点已经采用 UTC 并且已经具有 yyyy-MM-dd 的默认日期格式。如果您对此感到满意并且不想弄乱时区或格式,您也可以toString()它:

Instant instant = Instant.now();
instant.toString()
output: 2020-02-06T18:01:55.648475Z

不想要 T 和 Z?(Z 表示此日期是 UTC。Z 代表“Zulu”,又名“零时偏移”,又名 UTC):

instant.toString().replaceAll("[TZ]", " ")
output: 2020-02-06 18:01:55.663763

想要毫秒而不是纳秒?(所以您可以将其放入 sql 查询中):

instant.truncatedTo(ChronoUnit.MILLIS).toString().replaceAll("[TZ]", " ")
output: 2020-02-06 18:01:55.664

等等

【讨论】:

【参考方案6】:

或者,如果您仍想使用从模式创建的格式化程序 您可以只使用 LocalDateTime 而不是 Instant:

LocalDateTime datetime = LocalDateTime.now();
DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss").format(datetime)

【讨论】:

异常 java.time.temporal.UnsupportedTemporalTypeException:不支持的字段:YearOfEra OP:即时。答案:本地数据时间。经典【参考方案7】:
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy MM dd");
String text = date.toString(formatter);
LocalDate date = LocalDate.parse(text, formatter);

我相信这可能会有所帮助,您可能需要使用某种本地日期变体而不是即时

【讨论】:

这真的与 Instant 类无关。

以上是关于将 Instant 格式化为字符串的主要内容,如果未能解决你的问题,请参考以下文章

Jackson 将 ISO8601 格式的日期时间反序列化为 Java8 Instant

即使使用 jackson-datatype-jsr310,Instant 也无法序列化为适当的格式

将毫秒时间戳反序列化为 java.time.Instant

将“数字”字符串格式化为数字

如何将字符串号码格式化为美国电话号码格式

将字符串格式化为日期时间