格式化即时字符串

Dag*_*Dag 192 java datetime datetime-format java-8 java-time

我正在尝试使用新的java 8 time-api和模式将Instant格式化为String:

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

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

java.time.temporal.UnsupportedTemporalTypeException: Unsupported field: YearOfEra
    at java.time.Instant.getLong(Instant.java:608)
    at java.time.format.DateTimePrintContext.getValue(DateTimePrintContext.java:298)
    ...
Run Code Online (Sandbox Code Playgroud)

Jod*_*hen 257

时区

要格式化的Instant一个时区是必需的.如果没有时区,格式化程序将不知道如何将即时字段转换为人类日期时间字段,因此会抛出异常.

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

DateTimeFormatter formatter =
    DateTimeFormatter.ofLocalizedDateTime( FormatStyle.SHORT )
                     .withLocale( Locale.UK )
                     .withZone( ZoneId.systemDefault() );
Run Code Online (Sandbox Code Playgroud)

生成字符串

现在使用该格式化程序生成Instant的String表示形式.

Instant instant = Instant.now();
String output = formatter.format( instant );
Run Code Online (Sandbox Code Playgroud)

转储到控制台.

System.out.println("formatter: " + formatter + " with zone: " + formatter.getZone() + " and Locale: " + formatter.getLocale() );
System.out.println("instant: " + instant );
System.out.println("output: " + output );
Run Code Online (Sandbox Code Playgroud)

跑步时

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
Run Code Online (Sandbox Code Playgroud)

  • 谢谢!!BTW指向例如年份的"不支持的字段"的例外是非常钝的.也许这种情况应该_detected_并且应该抛出一个直接指向Instant中缺少的区域id的异常! (35认同)
  • 现在这给你 `yyyy-MM-dd hh:mm:ss` 格式:`DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss").withZone(ZoneId.of("Europe/Paris") ”));` (3认同)
  • @KorayTugay 因为虽然迂腐的“Instant 已经是 GMT”评论可能是正确的,但在面对抛出的异常跟踪时它们毫无帮助,因为在不指定时区的情况下格式化 Instant 是行不通的。如果格式化程序默认为 GMT 那就太好了,但是哦,好吧。 (2认同)

小智 29

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()));

}
Run Code Online (Sandbox Code Playgroud)

  • new Date().toInstant() 可以替换为 Instant.now() (3认同)
  • 尽管此代码可以回答问题,但提供有关“如何解决”和“为什么解决”问题的其他上下文将提高​​回答的长期价值。 (2认同)
  • 虽然此代码片段可以解决问题,但[包括解释](http://meta.stackexchange.com/questions/114762/explaining-entirely-‌​code-based-answers)确实有助于提高帖子的质量。请记住,您是在为将来的读者回答问题,而那些人可能不知道您建议代码的原因。 (2认同)

sto*_*ito 17

Instant类不包含区的信息,它只是从UNIX时代,即1070年1月1日从UTC毫秒存储时间戳.因此,格式化程序无法打印日期,因为日期始终打印为具体时区.您应该将时区设置为格式化程序,一切都会好的,如下所示:

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");
Run Code Online (Sandbox Code Playgroud)

  • 纳秒,而不是毫秒 (2认同)

Arc*_*ano 15

DateTimeFormatter.ISO_INSTANT.format(Instant.now())
Run Code Online (Sandbox Code Playgroud)

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

DateTimeFormatter.ISO_INSTANT.format(Instant.now().truncatedTo(ChronoUnit.SECONDS))
Run Code Online (Sandbox Code Playgroud)


cs_*_*pil 15

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

Instant instant = Instant.now();
instant.toString()
output: 2020-02-06T18:01:55.648475Z
Run Code Online (Sandbox Code Playgroud)


不想要 T 和 Z?(Z 表示这个日期是 UTC。Z 代表“祖鲁语”又名“零时差”又名 UTC):

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


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

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

等等。