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

ale*_*mix 10 java datetime rfc rfc3339

这似乎是一个具有许多不同答案的常见问题.在你回答之前,我已经使用了joda-timeatomdate,它们运行得很好.我对此感兴趣的不是使用哪个库,而是澄清了如何在java中定义RFC模式.


研究

根据我的理解和答案, RFC 3339是ISO 8601的简介.PHP 清楚地定义了RFC 3339日期时间模式Y-m-d\TH:i:sP.如果我们将这个定义转移到java 7(据我所知),我们最终会得到这个(在这个答案中也会提到):

// example "2005-08-15T15:52:01+00:00"
pattern = "yyyy-MM-dd'T'HH:mm:ssXXX";
Run Code Online (Sandbox Code Playgroud)

但是,像这样的一些堆栈溢出答案指向其中一个(或两者)作为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'";
Run Code Online (Sandbox Code Playgroud)

为了进一步复杂化,官方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";
Run Code Online (Sandbox Code Playgroud)

附注: Android不支持XXX时区模式,但您可以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";
Run Code Online (Sandbox Code Playgroud)

我的结论

与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"
};
Run Code Online (Sandbox Code Playgroud)

更新

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


要旨

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

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


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

Ada*_*ent 4

你基本上几乎回答了你自己的问题,除了你的要点并不适合所有情况......那就是它需要比你拥有的两个更多的模式(例如处理纳秒)。

这就是为什么 Joda 和 Java 8 有针对 ISO 8601(超集)的特殊解析器。

我知道您不需要引用其他库,但对于使用 Java 8 和/或想要明确限制为 RFC 3339 的其他库(我相信 joda iso 解析器将采用比 rfc 3339 更多的格式),有这个库:https ://github.com/ethlo/itu