使用 Joda Date & Time API 解析多种格式

我正在使用 Joda 解析包含日期/时间的第三方日志文件。日期/时间采用两种不同的格式之一,具体取决于我要解析的日志文件的年龄。

目前我有这样的代码:

try {
return DateTimeFormat.forPattern("yyyy/MM/dd HH:mm:ss").parseDateTime(datePart);
} catch (IllegalArgumentException e) {
return DateTimeFormat.forPattern("E, MMM dd, yyyy HH:mm").parseDateTime(datePart);
}

这种方法可行,但是违背了 Joshua Bloch 在有效 Java 第二版中的建议(第57条: 仅在异常情况下使用异常)。它还使得很难确定是否由于日志文件中的日期/时间错误而发生了 IllegalArgumentException。

您能建议一种不滥用异常的更好的方法吗?

32166 次浏览

不幸的是,我不相信 Joda Time 有这样的能力。如果有一个“ tryParseDateTime”方法就好了,但它并不存在。

我建议你将这种行为隔离到你自己的类中(类中有一个模式列表,并将依次尝试) ,这样丑陋就只存在于一个地方。如果这会导致性能问题,那么您可能需要尝试使用一些启发式方法来猜测首先尝试哪种格式。例如,在您的例子中,如果字符串以一个数字开头,那么它可能是第一个模式。

请注意,在 Joda Time,DateTimeFormatter通常是不可变的——你不应该在每次解析一行时都创建一个新的。创建它们一次并重用它们。

Joda-Time 通过允许指定多个解析器来支持这一点-DateTimeFormatterBuilder # append

只需使用生成器创建两个格式化程序,并对每个格式化程序调用 toParser()。然后使用构建器使用 append组合它们。

您可以使用 补充方法创建多个解析器并将它们添加到生成器:

DateTimeParser[] parsers = {
DateTimeFormat.forPattern( "yyyy-MM-dd HH" ).getParser(),
DateTimeFormat.forPattern( "yyyy-MM-dd" ).getParser() };
DateTimeFormatter formatter = new DateTimeFormatterBuilder().append( null, parsers ).toFormatter();


DateTime date1 = formatter.parseDateTime( "2010-01-01" );
DateTime date2 = formatter.parseDateTime( "2010-01-01 01" );