找到JDK7中日期的真正区别

Jay*_*ner 3 java datetime date jodatime date-difference

如何在JDK7中找到两个日期之间的差异,以便可以添加差异以使两个日期相等?

我尝试使用类似StackOverflow问题的解决方案,从另一个中减去一个日期的毫秒数,但这可能导致日期之间的差异不正确.

在这个例子中,我尝试通过设置使其a相等ba = (b-a) + a

private void makeTwoDatesEqual(GregorianCalendar a, GregorianCalendar b, DatatypeFactory datatypeFactory) {
    long b_minus_a = b.getTimeInMillis() - a.getTimeInMillis();
    Duration delta = datatypeFactory.newDuration(b_minus_a);
    delta.addTo(a);
}
Run Code Online (Sandbox Code Playgroud)

但结果出人意料:

之前

a = "2015-08-29T00:00:00.000Z"
b = "2040-01-01T00:00:00.000Z"
delta = "P24Y4M5DT0H0M0.000S" // 2 days longer than it should be.
Run Code Online (Sandbox Code Playgroud)

a = "2040-01-03T00:00:00.000Z"
b = "2040-01-01T00:00:00.000Z"
delta = "P24Y4M5DT0H0M0.000S"
a.equals(b) = false
Run Code Online (Sandbox Code Playgroud)

Joda-Time也有同样的问题:

private void makeTwoDatesEqual(GregorianCalendar a, GregorianCalendar b) {
    DateTime date1 = new DateTime(a);
    DateTime date2 = new DateTime(b);
    Interval interval = new Interval(a.getTimeInMillis(), b.getTimeInMillis());
    Period offsetPeriod = interval.toPeriod();
    date1 = date1.plus(offsetPeriod);
    date1.equals(date2); // false in my example
}
Run Code Online (Sandbox Code Playgroud)

之前

date1 = "2015-08-29T00:00:00.000-05:00"
date2 = "2040-01-01T00:00:00.000-05:00"
offsetPeriod = "P24Y4M2DT23H"
date1.equals(date2) = false
Run Code Online (Sandbox Code Playgroud)

date1 = "2039-12-31T23:00:00.000-05:00"
date2 = "2040-01-01T00:00:00.000-05:00"
offsetPeriod = "P24Y4M2DT23H"
date1.equals(date2) = false
Run Code Online (Sandbox Code Playgroud)

在被问到之前,应该没有必要考虑夏令时或闰年,因为在每个例子中,我都在相应的时间段内添加时间间隔.

Ole*_*.V. 7

ThreeTen Backport

有一些使用现代Java日期和时间API的可能性,称为JSR-310或java.time.你能用Java 7吗?当然!获取ThreeTen Backport并开始编码.:-)

日期时间之间的差异

编辑:如果你想要察觉你的ab时间点,换句话说,日期和时间,你可以使用一个适当的小时间单位的差异,或Duration类.两者都是一样的.由于Instants具有纳秒级精度,因此计算纳米级的差异将确保我们不会失去任何精度:

    Instant a = Instant.parse("2015-08-29T00:00:00.000Z");
    Instant b = Instant.parse("2040-01-01T00:00:00.000Z");
    long delta = ChronoUnit.NANOS.between(a, b);
    a = a.plusNanos(delta);
    System.out.println("a = " + a);
    System.out.println("b = " + b);
    System.out.println("delta = " + delta);
    System.out.println("a.equals(b) = " + a.equals(b));
Run Code Online (Sandbox Code Playgroud)

这打印

a = 2040-01-01T00:00:00Z
b = 2040-01-01T00:00:00Z
delta = 768182400000000000
a.equals(b) = true
Run Code Online (Sandbox Code Playgroud)

Duration课程的持续时间以小时,分钟,秒和秒为单位.您可以将它用于更长的持续时间,但它不知道几周,几个月或几年.

    Duration delta = Duration.between(a, b);
    a = a.plus(delta);
Run Code Online (Sandbox Code Playgroud)

否则和以前一样.这次输出是

a = 2040-01-01T00:00:00Z
b = 2040-01-01T00:00:00Z
delta = PT213384H
a.equals(b) = true
Run Code Online (Sandbox Code Playgroud)

所以差异是213384小时.只是因为在这种情况下分钟和秒数为0,它们不会被打印,否则它们将被打印; Duration也具有纳秒精度.

日期之间的差异

从字面上理解您的标题并注意到您的示例日期属于UTC午夜.如果您不关心一天中的时间而只想计算几天,几个月和几年,您可以使用Period课程或只计算天数.该Period课程适用于几天,几个月和几年的时间,并且具有1天的精确度(这意味着,不能用于小时和更小的单位).

    LocalDate a = LocalDate.of(2015, Month.AUGUST, 29);
    LocalDate b = LocalDate.of(2040, Month.JANUARY, 1);
    Period delta = Period.between(a, b);
    a = a.plus(delta);
Run Code Online (Sandbox Code Playgroud)

打印与以前一样,但现在打印

a = 2040-01-01
b = 2040-01-01
delta = P24Y4M3D
a.equals(b) = true
Run Code Online (Sandbox Code Playgroud)

计数天数就像上面的纳秒一样,只有天数:

    long delta = ChronoUnit.DAYS.between(a, b);
    a = a.plusDays(delta);
Run Code Online (Sandbox Code Playgroud)

这次输出包含:

delta = 8891
a.equals(b) = true
Run Code Online (Sandbox Code Playgroud)

请注意,因为月份的长度不一样,即使24年4个月3天似乎等于8891天,这往往不是确切的情况.这在@Meno Hochschild的评论Hugo的回答中有更详细的解释.因此,在Period一天和几天之间进行选择会产生影响,您应该根据更精确的要求进行选择.

结合期间和期限

有趣的是,Java不会在一段时间或几年,几天,几小时,几分钟和几秒的持续时间内提供课程.如果您愿意,可以使用Period年份,月份和日期的实例,然后使用Duration小时,分钟和秒的实例.它有点复杂,所以除非你说你需要它,否则我不会编写代码,但它是可行的.您当然也可以自己制定详细信息.

有人告诉我,ThreeTen-extra项目包括一个PeriodDuration结合了两者的课程.我自己没有任何经验.

  • 推荐ThreeTen Backport的建议.然而,问题在技术上是关于完整日期+时间(例如`2015-08-29T00:00:00.000Z`),因此更多的点对点答案将包括使用"Instant"和"Duration"的示例. (3认同)