将 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