Java SimpleDateFormat для часового пояса с разделителем двоеточий? - PullRequest
59 голосов
/ 04 марта 2010

У меня есть дата в следующем формате: 2010-03-01T00:00:00-08:00

Я добавил следующие SimpleDateFormats для его анализа:

private static final SimpleDateFormat[] FORMATS = {
        new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ"), //ISO8601 long RFC822 zone
        new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssz"), //ISO8601 long long form zone
        new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss"), //ignore timezone
        new SimpleDateFormat("yyyyMMddHHmmssZ"), //ISO8601 short
        new SimpleDateFormat("yyyyMMddHHmm"),
        new SimpleDateFormat("yyyyMMdd"), //birthdate from NIST IHE C32 sample
        new SimpleDateFormat("yyyyMM"),
        new SimpleDateFormat("yyyy") //just the year
    };

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

public static Date figureOutTheDamnDate(String wtf) {
    if (wtf == null) {
        return null;
    }
    Date retval = null;
    for (SimpleDateFormat sdf : FORMATS) {
        try {
            sdf.setLenient(false)
            retval = sdf.parse(wtf);
            System.out.println("Date:" + wtf + " hit on pattern:" + sdf.toPattern());
            break;
        } catch (ParseException ex) {
            retval = null;
            continue;
        }
    }

    return retval;
}

Кажется, что оно попало в шаблон yyyyMMddHHmm, но возвращает дату как Thu Dec 03 00:01:00 PST 2009.

Каков правильный шаблон для анализа этой даты?

ОБНОВЛЕНИЕ: мне НЕ НУЖЕН разбор часового пояса. Я не ожидаю возникновения проблем, чувствительных ко времени при перемещении между зонами, но как мне получить формат зоны "-08: 00" для разбора ????

Юнит-тест:

@Test
public void test_date_parser() {
    System.out.println("\ntest_date_parser");
    //month is zero based, are you effing kidding me
    Calendar d = new GregorianCalendar(2000, 3, 6, 13, 00, 00);
    assertEquals(d.getTime(), MyClass.figureOutTheDamnDate("200004061300"));
    assertEquals(new GregorianCalendar(1950, 0, 1).getTime(), MyClass.figureOutTheDamnDate("1950"));
    assertEquals(new GregorianCalendar(1997, 0, 1).getTime(),  MyClass.figureOutTheDamnDate("199701"));
    assertEquals(new GregorianCalendar(2010, 1, 25, 15, 19, 44).getTime(),   MyClass.figureOutTheDamnDate("20100225151944-0800"));

    //my machine happens to be in GMT-0800
    assertEquals(new GregorianCalendar(2010, 1, 15, 13, 15, 00).getTime(),MyClass.figureOutTheDamnDate("2010-02-15T13:15:00-05:00"));
    assertEquals(new GregorianCalendar(2010, 1, 15, 18, 15, 00).getTime(), MyClass.figureOutTheDamnDate("2010-02-15T18:15:00-05:00"));

    assertEquals(new GregorianCalendar(2010, 2, 1).getTime(), MyClass.figureOutTheDamnDate("2010-03-01T00:00:00-08:00"));
    assertEquals(new GregorianCalendar(2010, 2, 1, 17, 0, 0).getTime(), MyClass.figureOutTheDamnDate("2010-03-01T17:00:00-05:00"));
}

Вывод из модульного теста:

test_date_parser
Date:200004061300 hit on pattern:yyyyMMddHHmm
Date:1950 hit on pattern:yyyy
Date:199701 hit on pattern:yyyyMM
Date:20100225151944-0800 hit on pattern:yyyyMMddHHmmssZ
Date:2010-02-15T13:15:00-05:00 hit on pattern:yyyy-MM-dd'T'HH:mm:ss
Date:2010-02-15T18:15:00-05:00 hit on pattern:yyyy-MM-dd'T'HH:mm:ss
Date:2010-03-01T00:00:00-08:00 hit on pattern:yyyy-MM-dd'T'HH:mm:ss
Date:2010-03-01T17:00:00-05:00 hit on pattern:yyyy-MM-dd'T'HH:mm:ss

Ответы [ 11 ]

58 голосов
/ 04 марта 2010

JodaTime DateTimeFormat для спасения:

String dateString = "2010-03-01T00:00:00-08:00";
String pattern = "yyyy-MM-dd'T'HH:mm:ssZ";
DateTimeFormatter dtf = DateTimeFormat.forPattern(pattern);
DateTime dateTime = dtf.parseDateTime(dateString);
System.out.println(dateTime); // 2010-03-01T04:00:00.000-04:00

(разница во времени и часовых поясах в toString() только потому, что я нахожусь в GMT-4 и не указала локаль явно)

Если вы хотите получить java.util.Date, просто используйте DateTime#toDate():

Date date = dateTime.toDate();

Ожидание JDK7 ( JSR-310 ) JSR-310, ссылочная реализация называется ThreeTen (надеюсь, она перейдет в Java 8), если Вы хотите лучший форматтер в стандартном API Java SE. Текущий SimpleDateFormat действительно не ест двоеточие в обозначении часового пояса.

Обновление : в соответствии с обновлением вам явно не нужен часовой пояс. Это должно работать с SimpleDateFormat. Просто опустите его (Z) в шаблоне.

String dateString = "2010-03-01T00:00:00-08:00";
String pattern = "yyyy-MM-dd'T'HH:mm:ss";
SimpleDateFormat sdf = new SimpleDateFormat(pattern);
Date date = sdf.parse(dateString);
System.out.println(date); // Mon Mar 01 00:00:00 BOT 2010

(что соответствует моему часовому поясу)

33 голосов
/ 04 апреля 2014

Если бы вы использовали Java 7, вы могли бы использовать следующий шаблон даты и времени. Похоже, этот шаблон не поддерживается в более ранней версии Java.

String dateTimeString  = "2010-03-01T00:00:00-08:00";
DateFormat df = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX");
Date date = df.parse(dateTimeString);

Подробнее см. В документации SimpleDateFormat .

29 голосов
/ 20 января 2013

Вот фрагмент, который я использовал - с простым SimpleDateFormat. Надеюсь, что кто-то еще может извлечь из этого пользу:

public static void main(String[] args) {
    SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ") {
        public StringBuffer format(Date date, StringBuffer toAppendTo, java.text.FieldPosition pos) {
            StringBuffer toFix = super.format(date, toAppendTo, pos);
            return toFix.insert(toFix.length()-2, ':');
        };
    };
    // Usage:
    System.out.println(dateFormat.format(new Date()));
}

Выход:

- Usual Output.........: 2013-06-14T10:54:07-0200
- This snippet's Output: 2013-06-14T10:54:07-02:00

Или ... лучше, используйте более простой, другой шаблон:

SimpleDateFormat dateFormat2 = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX");
// Usage:
System.out.println(dateFormat2.format(new Date()));

Выход:

- This pattern's output: 2013-06-14T10:54:07-02:00

См. документы для этого .

13 голосов
/ 14 ноября 2013

Попробуйте это, его работа для меня:

Date date = javax.xml.bind.DatatypeConverter.parseDateTime("2013-06-01T12:45:01+04:00").getTime();

В Java 8:

OffsetDateTime dt = OffsetDateTime.parse("2010-03-01T00:00:00-08:00");
9 голосов
/ 19 апреля 2016

Если вы можете использовать JDK 1.7 или выше, попробуйте это:

public class DateUtil {
    private static SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX");

    public static String format(Date date) {
        return dateFormat.format(date);
    }

    public static Date parse(String dateString) throws AquariusException {
        try {
            return dateFormat.parse(dateString);
        } catch (ParseException e) {
            throw new AquariusException(e);
        }
    }
}

документ: https://docs.oracle.com/javase/7/docs/api/java/text/SimpleDateFormat.html который поддерживает новый формат часового пояса "XXX" (например, -3: 00)

В то время как JDK 1.6 поддерживает только другие форматы для часового пояса: «z» (например, NZST), «zzzz» (например, стандартное время Новой Зеландии), «Z» (например, +1200) и т. Д.

7 голосов
/ 19 апреля 2016

ТЛ; др

OffsetDateTime.parse( "2010-03-01T00:00:00-08:00" )

Подробнее

Ответ от BalusC правильный, но теперь устарел с Java 8.

java.time

Фреймворк java.time является преемником как библиотеки Joda-Time, так и старых проблемных классов даты и времени, связанных с ранними версиями Java (java.util.Date/.Calendar & java. text.SimpleDateFormat).

ISO 8601

Ваша строка входных данных соответствует стандарту ISO 8601 .

Классы java.time по умолчанию используют форматы ISO 8601 при разборе / генерации текстовых представлений значений даты и времени. Поэтому нет необходимости определять шаблон форматирования.

OffsetDateTime

Класс OffsetDateTime представляет момент на временной шкале, скорректированный до некоторого определенного смещения от UTC . По вашим данным, смещение составляет 8 часов после UTC, обычно используется на большей части западного побережья Северной Америки.

OffsetDateTime odt = OffsetDateTime.parse( "2010-03-01T00:00:00-08:00" );

Похоже, вам нужна только дата, в этом случае используйте класс LocalDate. Но имейте в виду, что вы отбрасываете данные, (а) время суток и (б) часовой пояс. Действительно, дата не имеет смысла без контекста часового пояса . В любой данный момент дата меняется по всему миру. Например, сразу после полуночи в Париже все еще «вчерашний день» в Монреале. Поэтому, хотя я предлагаю придерживаться значений даты и времени, вы можете легко конвертировать в LocalDate, если вы настаиваете.

LocalDate localDate = odt.toLocalDate();

часовой пояс

Если вы знаете предполагаемый часовой пояс, примените его. Часовой пояс - это смещение плюс правил, используемых для обработки аномалий, таких как Летнее время (DST) . Применение ZoneId дает нам объект ZonedDateTime.

ZoneId zoneId = ZoneId.of( "America/Los_Angeles" );
ZonedDateTime zdt = odt.atZoneSameInstant( zoneId );

Генерация строк

Чтобы сгенерировать строку в формате ISO 8601, позвоните toString.

String output = odt.toString();

Если вам нужны строки в других форматах, найдите переполнение стека для использования пакета java.util.format .

Преобразование в java.util.Date

Лучше всего избегать java.util.Date, но при необходимости вы можете конвертировать. Вызовите новые методы, добавленные к старым классам, например java.util.Date.from, где вы передаете Instant. Instant - это момент на временной шкале в UTC . Мы можем извлечь Instant из нашего OffsetDateTime.

java.util.Date utilDate = java.util.Date( odt.toInstant() );

О 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?

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

4 голосов
/ 24 мая 2013

Спасибо acdcjunior за ваше решение. Вот немного оптимизированная версия для форматирования и разбора :

public static final SimpleDateFormat XML_SDF = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ", Locale.FRANCE)
{
    private static final long serialVersionUID = -8275126788734707527L;

    public StringBuffer format(Date date, StringBuffer toAppendTo, java.text.FieldPosition pos)
    {            
        final StringBuffer buf = super.format(date, toAppendTo, pos);
        buf.insert(buf.length() - 2, ':');
        return buf;
    };

    public Date parse(String source) throws java.text.ParseException {
        final int split = source.length() - 2;
        return super.parse(source.substring(0, split - 1) + source.substring(split)); // replace ":" du TimeZone
    };
};
2 голосов
/ 17 ноября 2017

Вы можете использовать X в Java 7.

https://docs.oracle.com/javase/7/docs/api/java/text/SimpleDateFormat.html

static final SimpleDateFormat DATE_TIME_FORMAT = 
        new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

static final SimpleDateFormat JSON_DATE_TIME_FORMAT = 
        new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX");

private String stringDate = "2016-12-01 22:05:30";
private String requiredDate = "2016-12-01T22:05:30+03:00";

@Test
public void parseDateToBinBankFormat() throws ParseException {
    Date date = DATE_TIME_FORMAT.parse(stringDate);
    String jsonDate = JSON_DATE_TIME_FORMAT.format(date);

    System.out.println(jsonDate);
    Assert.assertEquals(jsonDate, requiredDate);
}
1 голос
/ 22 ноября 2016

Поскольку здесь отсутствует пример Apache FastDateFormat (щелкните для документации версий: 2.6 и 3.5 ), я добавляю один для тех, кому он может понадобиться. Ключевым здесь является шаблон ZZ (2 заглавные Z s).

import java.text.ParseException
import java.util.Date;
import org.apache.commons.lang3.time.FastDateFormat;
public class DateFormatTest throws ParseException {
    public static void main(String[] args) {
        String stringDateFormat = "yyyy-MM-dd'T'HH:mm:ssZZ";
        FastDateFormat fastDateFormat = FastDateFormat.getInstance(stringDateFormat);
        System.out.println("Date formatted into String:");
        System.out.println(fastDateFormat.format(new Date()));
        String stringFormattedDate = "2016-11-22T14:30:14+05:30";
        System.out.println("String parsed into Date:");
        System.out.println(fastDateFormat.parse(stringFormattedDate));
    }
}

Вот вывод кода:

Date formatted into String:
2016-11-22T14:52:17+05:30
String parsed into Date:
Tue Nov 22 14:30:14 IST 2016

Примечание. Приведенный выше код относится к языку Apache Commons 'lang3. Класс org.apache.commons.lang.time.FastDateFormat не поддерживает синтаксический анализ и поддерживает только форматирование. Например, вывод следующего кода:

import java.text.ParseException;
import java.util.Date;
import org.apache.commons.lang.time.FastDateFormat;
public class DateFormatTest {
    public static void main(String[] args) throws ParseException {
        String stringDateFormat = "yyyy-MM-dd'T'HH:mm:ssZZ";
        FastDateFormat fastDateFormat = FastDateFormat.getInstance(stringDateFormat);
        System.out.println("Date formatted into String:");
        System.out.println(fastDateFormat.format(new Date()));
        String stringFormattedDate = "2016-11-22T14:30:14+05:30";
        System.out.println("String parsed into Date:");
        System.out.println(fastDateFormat.parseObject(stringFormattedDate));
    }
}

будет таким:

Date formatted into String:
2016-11-22T14:55:56+05:30
String parsed into Date:
Exception in thread "main" java.text.ParseException: Format.parseObject(String) failed
    at java.text.Format.parseObject(Format.java:228)
    at DateFormatTest.main(DateFormatTest.java:12)
1 голос
/ 04 марта 2010

Попробуйте setLenient(false).

Приложение. Похоже, вы распознаете строки различного формата Date. Если вам нужно сделать запись, вам может понравиться этот пример , который расширяет InputVerifier.

...