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
而不是DD
。 DD
值是“一年中的一天”,我怀疑这是令人困惑的事情,基本上完全覆盖了月份部分。 有关更多详细信息,请参阅SimpleDateFormat
文档。
解析后打印出d1
和d2
可以validation出错了什么…
我想你应该看看joda-time图书馆http://joda-time.sourceforge.net/
Java的默认日期实现很糟糕,使用joda时间你可以计算你想要的间隔以及它与java Date类的兼容性。
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
, .Calendar
和java.text.SimpleDateFormat
。
现在处于维护模式的Joda-Time项目建议迁移到java.time。
要了解更多信息,请参阅Oracle教程 。 并搜索Stack Overflow以获取许多示例和解释。
许多java.timefunction在ThreeTen- Backport中反向移植到Java 6和7,并在ThreeTenABP中进一步适应Android (请参阅如何使用… )。
ThreeTen-Extra项目使用其他类扩展了java.time。 该项目是未来可能添加到java.time的试验场。 您可以在这里找到一些有用的课程,如Interval
, YearWeek
, YearQuarter
等。