Java:跨越几个月的两个日期之间的差异

我有以下代码成功地让我得到两天之间的差异(以天,小时,分钟,秒为单位):

SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-DD HH:mm:ss"); Date d1 = format.parse(startTime); Date d2 = format.parse(endTime); long diff = d2.getTime() - d1.getTime(); long diffSeconds = diff / 1000 % 60; long diffMinutes = diff / (60 * 1000) % 60; long diffHours = diff / (60 * 60 * 1000) % 24; long diffDays = diff / (24 * 60 * 60 * 1000); System.out.println("Start: " + startTime); System.out.println("End: " + endTime); System.out.println(diffDays + " days, " + diffHours + " hours, " + diffMinutes + " minutes, " + diffSeconds + " seconds"); 

但是,当日期跨越到另一个月时,这不起作用,例如:

 Start: 2013-07-31 10:15:01 End: 2013-08-01 11:22:33 -29 days, -22 hours, -52 minutes, -28 seconds Start: 2013-05-31 10:15:01 End: 2013-08-01 11:22:33 -29 days, -22 hours, -52 minutes, -28 seconds 

是否可以智能地跨越数月并获得准确的时差? 我对Joda很熟悉,但是我希望坚持使用标准的Java API,除非没有像Joda这样的东西是不可能的。

您正在使用

 SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-DD HH:mm:ss"); 

应该

 SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 

你会得到正确的结果

 Start: 2013-05-31 10:15:01 End: 2013-08-01 11:22:33 62 days, 1 hours, 7 minutes, 32 seconds 

这是日期模式的javadoc。 dd是每月一天。 DD是一年中的一天。 您的Date对象未按预期方式解析。 基本上, 31的日期值覆盖了月份值。

调试器是你的朋友。

虽然您应该将Joda Time简单地用作更好的日期/时间API,但我怀疑问题实际上只是您正在解析值不正确。 使用格式字符串:

 yyyy-MM-dd HH:mm:ss 

…用dd而不是DDDD值是“一年中的一天”,我怀疑这是令人困惑的事情,基本上完全覆盖了月份部分。 有关更多详细信息,请参阅SimpleDateFormat文档。

解析后打印出d1d2可以validation出错了什么…

我想你应该看看joda-time图书馆http://joda-time.sourceforge.net/

Java的默认日期实现很糟糕,使用joda时间你可以计算你想要的间隔以及它与java Date类的兼容性。

http://joda-time.sourceforge.net/quickstart.html (查看间隔和时间段)

TL;博士

在Java 9及更高版本中……

 Duration.between( LocalDateTime.parse( "2013-07-31T10:15:01" ) , LocalDateTime.parse( "2013-08-01T11:22:33" ) ).getDaysPart() 

…和…

 .getHoursPart() .getMinutesPart() .getSecondsPart() 

java.time

其他答案已过时。 Joda-Time项目现在处于维护模式,并建议迁移到Java内置的java.time类。

Period

Duration类表示未附加到时间轴的时间跨度,作为总秒数和纳秒数。 假设通用24小时工作日和60分钟工作小时,这样的跨度可以解释为天数,小时数,分钟数和秒数,因为不包括时区。

 LocalDateTime start = LocalDateTime.parse( "2013-07-31T10:15:01" ); LocalDateTime stop = LocalDateTime.parse( "2013-08-01T11:22:33" ); Duration d = Duration.between( start , stop ); 

span:2013-07-31T10:15:01 / 2013-08-01T11:22:33

d.toString():PT25H7M32S

这些字符串采用标准ISO 8601格式。 PT25H7M32S字符串使用PT25H7M32S格式,其中P标记开头, T将年 – 月 – 天与小时 – 分钟 – 秒分开。 字符串PT25H7M32S表示“二十五小时,七分钟和三十二秒”。

请在IdeOne.com中查看该代码 。

在Java 9及更高版本(但不是Java 8)中,这些Duration方法提取每个部分。

鉴于持续时间为49H30M20.123S ……

  • toDaysPart() = 2
  • toHoursPart() = 1
  • toMinutesPart() = 30
  • toSecondsPart() = 20
  • toMillisPart() = 123
  • toNanosPart() = toNanosPart()

关于java.time

java.time框架内置于Java 8及更高版本中。 这些类取代了麻烦的旧日期时间类,如java.util.Date.Calendarjava.text.SimpleDateFormat

现在处于维护模式的Joda-Time项目建议迁移到java.time。

要了解更多信息,请参阅Oracle教程 。 并搜索Stack Overflow以获取许多示例和解释。

许多java.timefunction在ThreeTen- Backport中反向移植到Java 6和7,并在ThreeTenABP中进一步适应Android (请参阅如何使用… )。

ThreeTen-Extra项目使用其他类扩展了java.time。 该项目是未来可能添加到java.time的试验场。 您可以在这里找到一些有用的课程,如IntervalYearWeekYearQuarter等。