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

时间:2013-08-15 15:48:58

标签: java date time

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

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这样的东西是不可能的。

4 个答案:

答案 0 :(得分:5)

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

yyyy-MM-dd HH:mm:ss

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

您可以在解析后打印d1d2来验证错误...

答案 1 :(得分:4)

您正在使用

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

Here's日期模式的javadoc。 dd是每月一次。 DD是一年中的一天。您的Date对象没有按照您的预期进行解析。基本上,31的日期值覆盖了月份值。

调试器是你的朋友。

答案 2 :(得分:1)

我认为你应该看一下joda-time library http://joda-time.sourceforge.net/

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

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

答案 3 :(得分:1)

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字符串使用的格式为PnYnMnDTnHnMnS,其中P标记开头,T将年 - 月 - 天与小时 - 分钟 - 秒分隔开来。字符串PT25H7M32S表示“二十五小时七分三十二秒”。

code live in IdeOne.com

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

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

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

关于java.time

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

现在位于Joda-Timemaintenance mode项目建议迁移到java.time。

要了解详情,请参阅Oracle Tutorial。并搜索Stack Overflow以获取许多示例和解释。

大部分java.time功能都被反向移植到Java 6& ThreeTen-Backport中的7,并进一步适应Android中的ThreeTenABP(见How to use…)。

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