Java SimpleDateFormat для часового пояса с разделителем двоеточий?
У меня свидание в следующем формате: 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 ответов
JodaTime 's 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
(что правильно в соответствии с моим часовым поясом)
Если бы вы использовали 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
документация
Вот фрагмент, который я использовал - с простым 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
Попробуйте это, его работа для меня:
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");
Если вы можете использовать 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) и т. Д.
ТЛ; др
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?
- Java SE 8, Java SE 9, Java SE 10 и более поздние версии
- Встроенный.
- Часть стандартного 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….
Проект ThreeTen-Extra расширяет java.time дополнительными классами. Этот проект является полигоном для возможных будущих дополнений к java.time. Вы можете найти некоторые полезные классы здесь, такие как Interval
, YearWeek
, YearQuarter
и многое другое.
Спасибо 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
};
};
Вы можете использовать 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);
}
Поскольку здесь отсутствует пример Apache FastDateFormat(щелкните для просмотра документации версий: 2.6 и 3.5), я добавляю пример для тех, кому он может понадобиться. Ключевым моментом здесь является шаблон ZZ
(2 заглавные Z
с).
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)
Пытаться setLenient(false)
,
Приложение: Похоже, что вы распознаете различные форматы Date
строки. Если вам нужно сделать запись, вы можете посмотреть на этот пример, который расширяет InputVerifier
,
Если строка даты похожа на 2018-07-20T12:18:29.802Z Используйте это
SimpleDateFormat fmt = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS'Z'");