gpt4 book ai didi

java - 在 JDK7 中找到日期之间的真正差异

转载 作者:塔克拉玛干 更新时间:2023-11-01 21:44:46 24 4
gpt4 key购买 nike

如何在 JDK7 中找到两个日期之间的差异,这样可以将差异加回去使两个日期相等?

我尝试使用来自 similar StackOverflow question 的解决方案, 从另一个日期中减去一个日期的毫秒数,但这会导致日期之间的差异不正确。

在此示例中,我尝试通过设置 a = (b-a) + a 使 a 等于 b

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);
}

但结果出乎意料:

之前

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

之后

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

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
}

之前

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

之后

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

在询问之前,不需要考虑夏令时或闰年,因为在每个示例中,我都将时间间隔加回与其派生的完全相同的时间段。

最佳答案

ThreeTen 向后移植

使用称为 JSR-310 或 java.time 的现代 Java 日期和时间 API 有多种可能性。您可以将其与 Java 7 一起使用吗?当然!得到 ThreeTen Backport并开始编码。

日期时间之间的差异

编辑:如果您想将ab 视为时间点,换句话说,日期和一天中的时间,您可以使用适当的小时间单位来表示差异,或者使用 Duration 类。两者的工作原理相同。由于 Instant 具有纳秒精度,计算纳秒级的差异将确保我们不会丢失任何精度:

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));

这打印

a = 2040-01-01T00:00:00Z
b = 2040-01-01T00:00:00Z
delta = 768182400000000000
a.equals(b) = true

Duration 类用于以小时、分钟、秒和秒的分数为单位的持续时间。您可以使用它更长时间,但它不知道几周、几个月或几年。

Duration delta = Duration.between(a, b);
a = a.plus(delta);

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

a = 2040-01-01T00:00:00Z
b = 2040-01-01T00:00:00Z
delta = PT213384H
a.equals(b) = true

所以相差 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);

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

a = 2040-01-01
b = 2040-01-01
delta = P24Y4M3D
a.equals(b) = true

计算天数就像上面的纳秒一样,只是用天代替:

long delta = ChronoUnit.DAYS.between(a, b);
a = a.plusDays(delta);

这次输出包含:

delta = 8891
a.equals(b) = true

请注意,由于月份的长度不同,即使 24 年 4 个月 3 天似乎等于 8891 天,但实际情况往往并非如此。这在 @Meno Hochschild’s comment 中有更详细的解释。和 Hugo’s answer .因此,Period 和天数之间的选择会有所不同,您应该根据更准确的要求进行选择。

组合期间和持续时间

Java 不提供年、月、日、小时、分钟和秒的时间段或持续时间的类,这有点滑稽。如果需要,您可以使用 Period 实例表示年、月和日,然后使用 Duration 实例表示小时、分钟和秒。有点复杂。

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

关于java - 在 JDK7 中找到日期之间的真正差异,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46244778/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com