兼容的String到java.util.Date

我试图将ISO 8601格式的字符串转换为java.util.Date。

如果使用Locale(比较样本),我发现模式“yyyy-MM-dd'T'HH:mm:ssZ”符合ISO8601标准。 但是,使用java.text.SimpleDateFormat,我无法转换格式正确的字符串“2010-01-01T12:00:00 + 01:00”。 我必须首先将它转换为“2010-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" ) ;

对我来说,这很好。


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

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

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

  • 有些时区的UTC时间不满,所以字符串不一定以“:00”结尾。
  • ISO8601只允许在时区中包含小时数,所以“+01”相当于“+01:00”
  • ISO8601允许使用“Z”来表示UTC而不是“+00:00”。
  • 更简单的解决方案可能是在JAXB中使用数据类型转换器,因为JAXB必须能够根据XML Schema规范解析ISO8601日期字符串。 如果您需要Date对象, javax.xml.bind.DatatypeConverter.parseDateTime("2010-01-01T12:00:00Z")将为您提供Calendar对象,并且您可以简单地使用它的getTime()。

    你也可以使用Joda-Time,但我不知道你为什么要这么做。


    好的,这个问题已经得到解答,但我仍然会放弃我的答案。 它可能有助于某人。

    我一直在寻找Android解决方案 (API 7)。

  • 乔达无法解决问题 - 这是巨大的,并且受到初始化缓慢的影响。 对于这个特定的目的来说,这似乎也是一个重大的过火。
  • 涉及javax.xml答案在Android API 7上不起作用。
  • 完成这个简单的课程。 它涵盖ISO 8601字符串的最常见形式 ,但在某些情况下(当您确信输入将采用此格式时)应该足够了。

    import java.text.ParseException;
    import java.text.SimpleDateFormat;
    import java.util.Calendar;
    import java.util.Date;
    import java.util.GregorianCalendar;
    
    /**
     * Helper class for handling a most common subset of ISO 8601 strings
     * (in the following format: "2008-03-01T13:00:00+01:00"). It supports
     * parsing the "Z" timezone, but many other less-used features are
     * missing.
     */
    public final class ISO8601 {
        /** Transform Calendar to ISO 8601 string. */
        public static String fromCalendar(final Calendar calendar) {
            Date date = calendar.getTime();
            String formatted = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")
                .format(date);
            return formatted.substring(0, 22) + ":" + formatted.substring(22);
        }
    
        /** Get current date and time formatted as ISO 8601 string. */
        public static String now() {
            return fromCalendar(GregorianCalendar.getInstance());
        }
    
        /** Transform ISO 8601 string to Calendar. */
        public static Calendar toCalendar(final String iso8601string)
                throws ParseException {
            Calendar calendar = GregorianCalendar.getInstance();
            String s = iso8601string.replace("Z", "+00:00");
            try {
                s = s.substring(0, 22) + s.substring(23);  // to get rid of the ":"
            } catch (IndexOutOfBoundsException e) {
                throw new ParseException("Invalid length", 0);
            }
            Date date = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ").parse(s);
            calendar.setTime(date);
            return calendar;
        }
    }
    

    性能说明:我每次都实例化新的SimpleDateFormat,以避免Android 2.1中的错误。 如果你和我一样惊讶,看看这个谜语。 对于其他Java引擎,您可以将实例缓存在私有静态字段中(使用ThreadLocal,以确保线程安全)。


    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 javadoc的示例部分找到更多示例。

    链接地址: http://www.djcxy.com/p/35109.html

    上一篇: compliant String to java.util.Date

    下一篇: How do I translate a ISO 8601 datetime string into a Python datetime object?