Недопустимый символ шаблона 'T' при разборе строки даты в java.util.Date

182

У меня есть строка с датой, и я хочу разобрать ее на обычную дату, используя java Date API, следующий код:

public static void main(String[] args) {
    String date="2010-10-02T12:23:23Z";
    String pattern="yyyy-MM-ddThh:mm:ssZ";
    SimpleDateFormat sdf=new SimpleDateFormat(pattern);
    try {
        Date d=sdf.parse(date);
        System.out.println(d.getYear());
    } catch (ParseException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

Однако я получил исключение: java.lang.IllegalArgumentException: Illegal pattern character 'T'

Так что мне интересно, придется ли мне разбивать строку и анализировать ее вручную?

Кстати, я попытался добавить символ одинарной кавычки по обе стороны от T:

String pattern="yyyy-MM-dd'T'hh:mm:ssZ";

Это тоже не работает.

hguser
источник
1
Вы знаете, что означает эта буква "Т"? Насколько я думаю, всю дату вместе с 'T' следует преобразовать в дату (не пропуская ее через «Одиночные кавычки», но нам не хватает шаблона разбора.
coding_idiot
9
TОзначает время и является частью ISO8601 стандарта для международного времени форматирования даты.
1
Странно. Упаковка Tв одинарные кавычки работала для меня (по крайней мере, с точки зрения решения ошибки разбора).
Agi

Ответы:

204

Обновление для Java 8 и выше

Теперь вы можете просто сделать Instant.parse("2015-04-28T14:23:38.521Z")и получить правильную вещь сейчас, тем более, что вам следует использовать Instantвзломанные java.util.Dateпоследние версии Java.

Вы должны использовать DateTimeFormatterвместо того, чтобы SimpleDateFormatter.

Оригинальный ответ:

Приведенное ниже объяснение остается в силе как то, что представляет формат. Но он был написан до того, как Java 8 стала повсеместной, поэтому он использует старые классы, которые не следует использовать, если вы используете Java 8 или выше.

Это работает с вводом с трейлингом, Zкак показано:

В схеме Tизбегается с 'обеих сторон.

Шаблон для Zконца в действительности XXXтакой же, как документированный в JavaDoc SimpleDateFormat, просто не очень понятно, как его использовать, так как он также Zявляется маркером для старой TimeZoneинформации.

Q2597083.java

import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.TimeZone;

public class Q2597083
{
    /**
     * All Dates are normalized to UTC, it is up the client code to convert to the appropriate TimeZone.
     */
    public static final TimeZone UTC;

    /**
     * @see <a href="http://en.wikipedia.org/wiki/ISO_8601#Combined_date_and_time_representations">Combined Date and Time Representations</a>
     */
    public static final String ISO_8601_24H_FULL_FORMAT = "yyyy-MM-dd'T'HH:mm:ss.SSSXXX";

    /**
     * 0001-01-01T00:00:00.000Z
     */
    public static final Date BEGINNING_OF_TIME;

    /**
     * 292278994-08-17T07:12:55.807Z
     */
    public static final Date END_OF_TIME;

    static
    {
        UTC = TimeZone.getTimeZone("UTC");
        TimeZone.setDefault(UTC);
        final Calendar c = new GregorianCalendar(UTC);
        c.set(1, 0, 1, 0, 0, 0);
        c.set(Calendar.MILLISECOND, 0);
        BEGINNING_OF_TIME = c.getTime();
        c.setTime(new Date(Long.MAX_VALUE));
        END_OF_TIME = c.getTime();
    }

    public static void main(String[] args) throws Exception
    {

        final SimpleDateFormat sdf = new SimpleDateFormat(ISO_8601_24H_FULL_FORMAT);
        sdf.setTimeZone(UTC);
        System.out.println("sdf.format(BEGINNING_OF_TIME) = " + sdf.format(BEGINNING_OF_TIME));
        System.out.println("sdf.format(END_OF_TIME) = " + sdf.format(END_OF_TIME));
        System.out.println("sdf.format(new Date()) = " + sdf.format(new Date()));
        System.out.println("sdf.parse(\"2015-04-28T14:23:38.521Z\") = " + sdf.parse("2015-04-28T14:23:38.521Z"));
        System.out.println("sdf.parse(\"0001-01-01T00:00:00.000Z\") = " + sdf.parse("0001-01-01T00:00:00.000Z"));
        System.out.println("sdf.parse(\"292278994-08-17T07:12:55.807Z\") = " + sdf.parse("292278994-08-17T07:12:55.807Z"));
    }
}

Производит следующий вывод:

sdf.format(BEGINNING_OF_TIME) = 0001-01-01T00:00:00.000Z
sdf.format(END_OF_TIME) = 292278994-08-17T07:12:55.807Z
sdf.format(new Date()) = 2015-04-28T14:38:25.956Z
sdf.parse("2015-04-28T14:23:38.521Z") = Tue Apr 28 14:23:38 UTC 2015
sdf.parse("0001-01-01T00:00:00.000Z") = Sat Jan 01 00:00:00 UTC 1
sdf.parse("292278994-08-17T07:12:55.807Z") = Sun Aug 17 07:12:55 UTC 292278994

источник
26

ТЛ; др

Используйте java.time.Instantкласс для разбора текста в стандартном формате ISO 8601, представляющего момент в UTC.

Instant.parse( "2010-10-02T12:23:23Z" )

ISO 8601

Этот формат определяется стандартом ISO 8601 для строковых форматов даты и времени.

Обе:

… Использовать форматы ISO 8601 по умолчанию для анализа и генерации строк.

Как правило, вам следует избегать использования старых классов java.util.Date /.Calendar & java.text.SimpleDateFormat, поскольку они заведомо хлопотны, сбивают с толку и имеют недостатки. Если требуется для взаимодействия, вы можете конвертировать туда и сюда.

java.time

В Java 8 и более поздние версии встроен новый фреймворк java.time . Вдохновленный Joda-Time , определенным JSR 310 и расширенным проектом ThreeTen-Extra .

Instant instant = Instant.parse( "2010-10-02T12:23:23Z" );  // `Instant` is always in UTC.

Преобразовать в старый класс.

java.util.Date date = java.util.Date.from( instant );  // Pass an `Instant` to the `from` method.

Часовой пояс

При необходимости вы можете назначить часовой пояс.

ZoneId zoneId = ZoneId.of( "America/Montreal" ); // Define a time zone rather than rely implicitly on JVM’s current default time zone.
ZonedDateTime zdt = ZonedDateTime.ofInstant( instant , zoneId );  // Assign a time zone adjustment from UTC.

Перерабатывать.

java.util.Date date = java.util.Date.from( zdt.toInstant() );  // Extract an `Instant` from the `ZonedDateTime` to pass to the `from` method.

Joda времени

ОБНОВЛЕНИЕ: проект Joda-Time сейчас находится в режиме обслуживания. Команда советует перейти на классы java.time .

Вот пример кода в Joda-Time 2.8.

org.joda.time.DateTime dateTime_Utc = new DateTime( "2010-10-02T12:23:23Z" , DateTimeZone.UTC );  // Specifying a time zone to apply, rather than implicitly assigning the JVM’s current default.

Преобразовать в старый класс. Обратите внимание, что назначенный часовой пояс теряется при конвертации, поскольку juDate нельзя назначить часовой пояс.

java.util.Date date = dateTime_Utc.toDate(); // The `toDate` method converts to old class.

Часовой пояс

При необходимости вы можете назначить часовой пояс.

DateTimeZone zone = DateTimeZone.forID( "America/Montreal" );
DateTime dateTime_Montreal = dateTime_Utc.withZone ( zone );

Таблица типов даты и времени в Java, как современных, так и устаревших.


О java.time

Java.time каркас встроен в Java 8 и более поздних версий. Эти классы вытеснять неприятные старые устаревшие классы даты и времени , такие как java.util.Date, Calendar, и SimpleDateFormat.

Проект Joda-Time , находящийся сейчас в режиме обслуживания , рекомендует перейти на классы java.time .

Чтобы узнать больше, смотрите Oracle Tutorial . И поиск переполнения стека для многих примеров и объяснений. Спецификация JSR 310 .

Вы можете обмениваться объектами java.time напрямую с вашей базой данных. Используйте драйвер JDBC, соответствующий JDBC 4.2 или более поздней версии . Нет необходимости в строках, нет необходимости в java.sql.*классах.

Где взять классы java.time?

  • Java SE 8 , Java SE 9 и более поздние
    • Встроенный.
    • Часть стандартного Java API со встроенной реализацией.
    • Java 9 добавляет некоторые незначительные функции и исправления.
  • Java SE 6 и Java SE 7
    • Большая часть функциональности java.time перенесена на Java 6 и 7 в ThreeTen-Backport .
  • Android
    • Более поздние версии Android связывают реализации классов java.time.
    • Для более ранних версий Android (<26) проект ThreeTenABP адаптирует ThreeTen-Backport (упомянутый выше). Смотрите Как использовать ThreeTenABP… .

Таблица какой библиотеки java.time использовать с какой версией Java или Android.

Проект ThreeTen-Extra расширяет java.time дополнительными классами. Этот проект является полигоном для возможных будущих дополнений к java.time. Вы можете найти некоторые полезные классы здесь , такие как Interval, YearWeek, YearQuarter, и более .

Базилик Бурк
источник
1
@AalapPatel Помните, что проект Joda-Time сейчас находится в режиме обслуживания. Команда советует перейти на классы java.time. Оба усилия возглавляются одним и тем же человеком, Стивеном Коулборном.
Василий Бурк
Спасибо, я использую его, чтобы получить миллисекунды с сервера, который вернул время UTC и / или локаль, и в этом случае хорошо работает для меня ..
Aalap Patel
Instant.parse нужен минимальный уровень API 26 в Android
Naveed Ahmad
1
@NaveedAhmad См. Маркеры внизу Ответа о проектах ThreeTen- Backport и ThreeTenABP .
Василий Бурк
0

На данный момент есть два вышеупомянутых ответа, и оба они длинные (и tl; dr слишком короткое ИМХО), поэтому я пишу сводку по своему опыту, начав использовать новую библиотеку java.time (применимо, как отмечено в других ответах к версии Java). 8+). ISO 8601 устанавливает стандартный способ записи даты: YYYY-MM-DDформат даты и времени только такой, как показано ниже (может быть 0, 3, 6 или 9 цифр в миллисекундах), и строка форматирования не требуется:

import java.time.Instant;
public static void main(String[] args) {
    String date="2010-10-02T12:23:23Z";
    try {
        Instant myDate = Instant.parse(date);
    } catch (ParseException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

Мне это не нужно, но поскольку получение года в коде из вопроса, то:
это сложнее, не может быть сделано Instantнапрямую, может быть выполнено с помощью Calendarвопросов. Получить целое значение текущего года в Java и преобразовать Java. время для календаря, но IMHO, поскольку формат является фиксированной подстрокой, более простой в использовании:

myDate.toString().substring(0,4);
Алексей Мартианов
источник