SimpleDateFormat错误地解析字符串

时间:2014-04-24 20:55:22

标签: java simpledateformat java.util.date java.util.calendar

String s = 19.17.38.008000;
DateFormat f = new SimpleDateFormat("HH.mm.ss.SSSSSS");
Date d = f.parse(s);
system.out.println(d);

这是我正在运行的代码它运行正常,除非它打印时打印时间 19点17分46秒。请有人向我解释

作为旁注:

String s = 19.17.38.008000;
DateFormat f = new SimpleDateFormat("HH.mm.ss");
Date d = f.parse(s);
system.out.println(d);

此代码将正确打印相同的字符串减去毫秒数。有人请告诉我这里缺少什么。

编辑:感谢您的回答我认为这里的问题是我正在读取38.008000为.008秒,但sdf正在读取SSS为8000毫秒,这不是一回事。

3 个答案:

答案 0 :(得分:5)

SimpleDateFormat类将008000解释为8000毫秒或8秒,并将其添加到已解释的38秒。

如果我们有这个:

String s = "19.17.38.009000";

然后我们会得到这个输出,加上9秒:

Thu Jan 01 19:17:47 PST 1970

从字符串末尾删除3个额外的零。如果有6位数,那么它们看起来应该代表微秒(百万分之一秒),而不是毫秒(千分之一秒)。

String s = "19.17.38.008";

输出:

Thu Jan 01 19:17:38 PST 1970

答案 1 :(得分:3)

SSSSSS即使你放了6个也仍然是毫秒。 19:17:38 + 008000 ms是19:17:46所以它是正确的,如果令人惊讶的话。

AFAIK Java 8中的java.time库支持微秒(和纳秒)时间戳。

感谢@Meno的更正。

答案 2 :(得分:0)

SimpleDateFormat does not parse beyond milliseconds precision。另外,请查看 documentation(强调我的)中的以下描述:

<块引用>

从给定字符串的开头解析文本以生成日期。 该方法可能不会使用给定字符串的整个文本

因此,即使您的模式有六个 SSimpleDateFormat 也仅使用其中的前三个,即

HH.mm.ss.SSS

换句话说,对于您的字符串,无论您使用 SimpleDateFormat 还是 HH.mm.ss.SSSHH.mm.ss.SSSSSS 都会给您同样的错误结果。 SimpleDateFormat 充满了许多这样的惊喜。为了解决使用 SimpleDateFormat 时的问题,您需要从字符串末尾删除三个 0,即如果您解析 19.17.38.008SimpleDateFormat 会给您的想要的结果。

因此,旧的日期时间 API(java.util 日期时间类型及其格式 API,SimpleDateFormat)不仅过时而且容易出错。建议完全停止使用它们并切换到 java.timemodern date-time API*

使用现代日期时间 API 的演示:

import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.util.Locale;

public class Main {
    public static void main(String args[]) {
        String s = "19.17.38.008000";
        DateTimeFormatter dtf = DateTimeFormatter.ofPattern("HH.mm.ss.SSSSSS", Locale.ENGLISH);
        LocalTime time = LocalTime.parse(s, dtf);
        System.out.println(time);
    }
}

输出:

19:17:38.008

请注意,如果您尝试使用 HH.mm.ss.SSS,对于您的字符串,使用 java.time API,它会抛出一个异常,提醒您有问题,而 SimpleDateFormat 会默默地解析您的字符串,给出你这是一个不受欢迎的结果。

modern date-time API 中了解有关 Trail: Date Time* 的更多信息。


* 出于任何原因,如果您必须坚持使用 Java 6 或 Java 7,您可以使用 ThreeTen-Backport,它将大部分 java.time 功能向后移植到 Java 6 & 7. 如果您正在为 Android 项目工作并且您的 Android API 级别仍然不符合 Java-8,请检查 Java 8+ APIs available through desugaringHow to use ThreeTenABP in Android Project