应该使用什么模式来解析java中的RFC 3339日期时间字符串

时间:2016-11-01 21:57:35

标签: java datetime rfc rfc3339

这似乎是一个有许多不同答案的常见问题。在您回答之前,我使用了joda-timeatomdate,但效果很好。我对此感兴趣的不是使用什么库,而是澄清了如何在java中定义RFC模式。

研究

根据我的理解和this answer RFC 3339是ISO 8601的配置文件.PHP clearly defines RFC 3339日期时间模式为Y-m-d\TH:i:sP。如果我们将这个定义转移到java 7(据我所知),我们最终会得到这个(在this answer中也会提到):

// example "2005-08-15T15:52:01+00:00"
pattern = "yyyy-MM-dd'T'HH:mm:ssXXX";

然而,像this one这样的几个堆栈溢出答案指向其中一个(或两者)作为RFC 3339的正确模式

// example "2016-11-01T20:44:39Z"
pattern = "yyyy-MM-dd'T'HH:mm:ss'Z'";

// example "1937-01-01T12:00:27.87Z"
pattern = "yyyy-MM-dd'T'HH:mm:ss.SSS'Z'";

为了使问题更加复杂,官方RFC 3339文档列出了以下所有这些示例(我添加了我认为的相应模式):

// 1996-12-19T16:39:57-08:00
pattern = "yyyy-MM-dd'T'HH:mm:ssXXX";

// 1990-12-31T23:59:60Z
pattern = "yyyy-MM-dd'T'HH:mm:ss'Z'";

// 1990-12-31T15:59:60-08:00
pattern = "yyyy-MM-dd'T'HH:mm:ssXXX";

// 1937-01-01T12:00:27.87+00:20
pattern = "yyyy-MM-dd'T'HH:mm:ss.SSSXXX";

附注: Android不支持时区的XXX模式,但您可以按照this answer使用ZZZZZ

我认为让我感到困惑的部分原因是我总是看到RFC 822和RFC 2822各有一个模式特别提到,所以我假设RFC 3339也可以归结为单一模式匹配:

static String RFC_822 = "EEE, dd MMM yy HH:mm:ss zzz";
static String RFC_2822 = "EEE, dd MMM yyyy HH:mm:ss zzz";

我的结论

与php不同,RFC 3339不能仅使用单个匹配表达式在java中表示。相反,所有这些都是有效的RFC 3339模式,并且必须在通过SimpleDateFormat解析日期时间字符串时进行检查:

static String[] RFC_3339_VARIANTS = {
        "yyyy-MM-dd'T'HH:mm:ss'Z'",
        "yyyy-MM-dd'T'HH:mm:ss.SSS'Z'",
        "yyyy-MM-dd'T'HH:mm:ssXXX",
        "yyyy-MM-dd'T'HH:mm:ss.SSSXXX"
};

更新

使问题复杂化SimpleDateFormat似乎无法正确处理'Z'时区文字。它不是假设UTC,而是默认为PST或您当地时间(我不确定哪个)。这意味着你可能需要用+00:00手动替换'Z'文字来纠正这种行为?

主旨

正如所建议的,我创建了一个实用程序类Gist,其中包含我当前运行的代码。这应该在Android上运行,并且与Java 7+兼容。请随时提出任何问题或发表评论。如果有足够的兴趣,我可以将其转移到Github,以便其他人可以贡献:

https://gist.github.com/oseparovic/d9ee771927ac5f3aefc8ba0b99c0cf38

我是否正确理解这一点,还是我完全脱离了?我非常感谢你们提供的关于如何在java 7中解析RFC 3339字符串的任何澄清。

1 个答案:

答案 0 :(得分:1)

你基本上几乎已经回答了你自己的问题,除非你的要点在所有情况下都不正确......也就是说它需要比你拥有的更多的模式(例如处理纳秒)。

这就是Joda和Java 8为ISO 8601(超集)提供特殊解析器的原因。

我知道你不需要引用其他库,但是对于那些使用Java 8并且或者想明确限制为RFC 3339的其他人(joda iso解析器,我相信它将采用比rfc 3339更多的格式),这就是图书馆:https://github.com/ethlo/itu