319

我正在尝试使用新的 Java 8 日期和时间 API 和以下模式将 Instant 格式化为字符串:

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)
    ...
4

7 回答 7

444

时区

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

时区可以直接添加到格式化程序使用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
于 2014-12-15T11:48:50.913 回答
64
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()));

}
于 2018-02-03T10:50:11.907 回答
44
DateTimeFormatter.ISO_INSTANT.format(Instant.now())

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

DateTimeFormatter.ISO_INSTANT.format(Instant.now().truncatedTo(ChronoUnit.SECONDS))
于 2019-03-07T22:58:24.833 回答
26

该类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");
于 2014-12-12T16:27:38.310 回答
19

瞬间已经采用 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

等等

于 2020-02-06T18:32:02.683 回答
0

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

LocalDateTime datetime = LocalDateTime.now();
DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss").format(datetime)
于 2018-11-16T17:38:50.937 回答
-3
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy MM dd");
String text = date.toString(formatter);
LocalDate date = LocalDate.parse(text, formatter);

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

于 2014-08-10T13:50:34.520 回答