Відформатуйте миттєве до рядка


227

Я намагаюсь відформатувати миттєвий рядок за допомогою нового 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)
    ...

Відповіді:


310

Часовий пояс

Для форматування часового поясу потрібно. Без часового поясу форматор не знає, як перетворити мить у людські поля дати та часу, і тому кидає виняток.Instant

Часовий пояс можна додати безпосередньо до форматера за допомогою withZone().

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

Створення рядка

Тепер використовуйте цей форматтер, щоб створити String представлення вашого Миттєвого.

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

50
Дякую!! До речі, виняток "непідтримуване поле", що вказує, наприклад, на рік, є надзвичайно тупим. Можливо, цю ситуацію слід виявити, і виключення, що вказує безпосередньо на пропущений ідентифікатор зони в Миттєвому миті, має бути кинуто!
davidbak

1
Більш химерно, якщо ви включаєте .withZone (наприклад, .withZone (ZoneId.of ("Z")) і форматуєте LocalDateTime, зона ІГНОРОВА! Доки включено .withZone (), той же формат може використовуватися як для Instant, так і для LocalDateTime, не впливаючи на час, показаний для останнього.
Гленн

1
Чому так важко прийняти той факт, що в системі Instant є TimeZone, яка є GMT?
Корай Тугай

1
@KorayTugay Оскільки коментарі "Миттєвий вже є GMT", коментарі можуть бути правдивими, якщо вони стикаються з викинутим трактом винятку, оскільки форматування Миттєвого сигналу без визначення часового поясу не працює. Було б добре, якби формат замовчував GMT, але добре.
Ti Strga

Тепер це дає вам yyyy-MM-dd hh:mm:ssформат:DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss").withZone(ZoneId.of("Europe/Paris"));
WesternGun

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

}

3
Хоча цей код може відповісти на питання, надаючи додатковий контекст щодо того, як і чому він вирішує проблему, покращить довгострокове значення відповіді.
Олександр

1
Хоча цей фрагмент коду може вирішити питання, у тому числі пояснення дійсно допомагає покращити якість вашої публікації. Пам'ятайте, що ви відповідаєте на запитання читачів у майбутньому, і ці люди можуть не знати причини вашої пропозиції щодо коду.
Розаріо Перейра Фернандес

23
DateTimeFormatter.ISO_INSTANT.format(Instant.now())

Це позбавить вас від необхідності конвертувати в UTC. Однак деякі часові рамки інших мов можуть не підтримувати мілісекунди, так що ви повинні зробити

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

Що ви маєте на увазі під "часовими рамками деяких інших мов"? Чи автоматично ISO_INSTANT.format () автоматично скорочується до секунд?
Гуантонг Шень

Деякі рамки очікують, що час підняться лише на секунди, оскільки вони не дотримуються повної конвенції ISO і розриваються, коли у вхідному рядку є мілісекунди.
Архімед

21

InstantКлас не містить інформації про зоні, він тільки зберігає тимчасові мітки в мілісекундах від UNIX епохи, тобто 1 січня 1070 від 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");

4

Або якщо ви все ще хочете використовувати формат, створений за шаблоном, ви можете просто використовувати LocalDateTime замість миттєвого:

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

4

Факти вже в UTC і вже мають формат дати за замовчуванням yyyy-MM-dd . Якщо ви цим задоволені і не хочете возитися з часовими поясами чи форматуванням, ви також можете toString():

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


Не хочете T і Z? (Z позначає, що ця дата є UTC. Z позначає "зулу", також "нульовий час зміщення" ака 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

тощо.


-3
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy MM dd");
String text = date.toString(formatter);
LocalDate date = LocalDate.parse(text, formatter);

Я вважаю, що це може допомогти, можливо, вам доведеться використовувати якусь варіацію localdate замість миттєвої

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.