Недопустимый символ шаблона 'T' при разборе строки даты в java.util.Date - PullRequest
154 голосов
/ 08 апреля 2010

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

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";

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

Ответы [ 2 ]

175 голосов
/ 08 апреля 2010

Обновление для 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
20 голосов
/ 22 сентября 2014

ТЛ; др

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.

Convert.

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

Joda-Time

ОБНОВЛЕНИЕ: проект 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.

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

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

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

...