将符合 ISO 8601 标准的字符串转换为 java.util.Date

2022-08-31 04:03:55

我正在尝试将ISO 8601格式的字符串转换为.java.util.Date

我发现如果与区域设置(比较示例)一起使用,则该模式符合ISO8601标准。yyyy-MM-dd'T'HH:mm:ssZ

但是,使用,我无法转换格式正确的字符串 。我必须首先将其转换为 ,不带冒号。java.text.SimpleDateFormat2010-01-01T12:00:00+01:002010-01-01T12:00:00+0100

因此,目前的解决方案是

SimpleDateFormat ISO8601DATEFORMAT = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ", Locale.GERMANY);
String date = "2010-01-01T12:00:00+01:00".replaceAll("\\+0([0-9]){1}\\:00", "+0$100");
System.out.println(ISO8601DATEFORMAT.parse(date));

这显然不是那么好。是我错过了什么,还是有更好的解决方案?


多亏了JuanZe的评论,我发现了Joda-Time的魔力,这里也描述了它。

因此,解决方案是

DateTimeFormatter parser2 = ISODateTimeFormat.dateTimeNoMillis();
String jtdate = "2010-01-01T12:00:00+01:00";
System.out.println(parser2.parseDateTime(jtdate));

或者更简单地说,通过构造函数使用默认解析器:

DateTime dt = new DateTime( "2010-01-01T12:00:00+01:00" ) ;

对我来说,这很好。


答案 1

不幸的是,SimpleDateFormat(Java 6及更早版本)可用的时区格式不符合ISO 8601标准。SimpleDateFormat理解时区字符串,如“GMT + 01:00”或“+0100”,后者根据RFC # 822

即使 Java 7 根据 ISO 8601 添加了对时区描述符的支持,SimpleDateFormat 仍然无法正确解析完整的日期字符串,因为它不支持可选部件。

使用正则表达式重新格式化输入字符串当然是一种可能性,但替换规则并不像您的问题中那么简单:

  • 某些时区不是 UTC 的整整一小时,因此字符串不一定以“:00”结尾。
  • ISO8601 只允许时区中包含的小时数,因此“+01”相当于“+01:00”
  • ISO8601 允许使用“Z”来表示 UTC,而不是“+00:00”。

更简单的解决方案可能是在 JAXB 中使用数据类型转换器,因为 JAXB 必须能够根据 XML 模式规范解析 ISO8601 日期字符串。 会给你一个对象,如果你需要一个对象,你可以简单地使用它上的getTime()。javax.xml.bind.DatatypeConverter.parseDateTime("2010-01-01T12:00:00Z")CalendarDate

您可能也可以使用Joda-Time,但我不知道为什么您应该为此烦恼(Update 2022;也许是因为Android的软件包中缺少整个部分)。javax.xml.bindjavax.xml


答案 2

Java 7 文档所祝福的方式:

DateFormat df1 = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSZ");
String string1 = "2001-07-04T12:08:56.235-0700";
Date result1 = df1.parse(string1);

DateFormat df2 = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSXXX");
String string2 = "2001-07-04T12:08:56.235-07:00";
Date result2 = df2.parse(string2);

您可以在 SimpleDateFormat javadocExamples 部分找到更多示例。

UPD 02/13/2020:在Java 8中有一种全新的方法来做到这一点。