Java 8:两个 LocalDateTime 在多个单位中的差异

2022-08-31 04:47:35

我正在尝试计算两个之间的差异。LocalDateTime

输出的格式需要为 。这是我写的:y years m months d days h hours m minutes s seconds

import java.time.Duration;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.Period;
import java.time.ZoneId;

public class Main {

    static final int MINUTES_PER_HOUR = 60;
    static final int SECONDS_PER_MINUTE = 60;
    static final int SECONDS_PER_HOUR = SECONDS_PER_MINUTE * MINUTES_PER_HOUR;

    public static void main(String[] args) {
        LocalDateTime toDateTime = LocalDateTime.of(2014, 9, 9, 19, 46, 45);
        LocalDateTime fromDateTime = LocalDateTime.of(1984, 12, 16, 7, 45, 55);

        Period period = getPeriod(fromDateTime, toDateTime);
        long time[] = getTime(fromDateTime, toDateTime);

        System.out.println(period.getYears() + " years " + 
                period.getMonths() + " months " + 
                period.getDays() + " days " +
                time[0] + " hours " +
                time[1] + " minutes " +
                time[2] + " seconds.");


    }

    private static Period getPeriod(LocalDateTime dob, LocalDateTime now) {
        return Period.between(dob.toLocalDate(), now.toLocalDate());
    }

    private static long[] getTime(LocalDateTime dob, LocalDateTime now) {
        LocalDateTime today = LocalDateTime.of(now.getYear(),
                now.getMonthValue(), now.getDayOfMonth(), dob.getHour(), dob.getMinute(), dob.getSecond());
        Duration duration = Duration.between(today, now);

        long seconds = duration.getSeconds();

        long hours = seconds / SECONDS_PER_HOUR;
        long minutes = ((seconds % SECONDS_PER_HOUR) / SECONDS_PER_MINUTE);
        long secs = (seconds % SECONDS_PER_MINUTE);

        return new long[]{hours, minutes, secs};
    }
}

我得到的输出是 。我已经从这个网站检查了我的结果(带有值和)。以下屏幕截图显示了输出:29 years 8 months 24 days 12 hours 0 minutes 50 seconds12/16/1984 07:45:5509/09/2014 19:46:45

Epoch Converter

我很确定月份值后面的字段来自我的代码。任何建议都会非常有帮助。

更新

我已经从另一个网站测试了我的结果,我得到的结果是不同的。这里是:计算两个日期之间的持续时间(结果:29年,8个月,24天,12小时,0分钟和50秒)。

更新

由于我从两个不同的站点获得了两个不同的结果,我想知道我的计算算法是否合法。如果我使用以下两个对象:LocalDateTime

LocalDateTime toDateTime = LocalDateTime.of(2014, 9, 10, 6, 40, 45);
LocalDateTime fromDateTime = LocalDateTime.of(1984, 12, 16, 7, 45, 55);

然后输出来了:29 years 8 months 25 days -1 hours -5 minutes -10 seconds.

从这个链接应该是.因此,该算法也需要处理负数。29 years 8 months 24 days 22 hours, 54 minutes and 50 seconds

请注意,问题不在于哪个网站给了我什么结果,我需要知道正确的算法,需要有正确的结果。


答案 1

我发现最好的方法是使用ChronoUnit。

long minutes = ChronoUnit.MINUTES.between(fromDate, toDate);
long hours = ChronoUnit.HOURS.between(fromDate, toDate);

其他文档在这里: https://docs.oracle.com/javase/tutorial/datetime/iso/period.html


答案 2

不幸的是,似乎没有跨越时间的周期类,因此您可能必须自己进行计算。

幸运的是,日期和时间类有很多实用方法,可以在一定程度上简化它。以下是计算差值的方法,尽管不一定是最快的:

LocalDateTime fromDateTime = LocalDateTime.of(1984, 12, 16, 7, 45, 55);
LocalDateTime toDateTime = LocalDateTime.of(2014, 9, 10, 6, 40, 45);

LocalDateTime tempDateTime = LocalDateTime.from( fromDateTime );

long years = tempDateTime.until( toDateTime, ChronoUnit.YEARS );
tempDateTime = tempDateTime.plusYears( years );

long months = tempDateTime.until( toDateTime, ChronoUnit.MONTHS );
tempDateTime = tempDateTime.plusMonths( months );

long days = tempDateTime.until( toDateTime, ChronoUnit.DAYS );
tempDateTime = tempDateTime.plusDays( days );


long hours = tempDateTime.until( toDateTime, ChronoUnit.HOURS );
tempDateTime = tempDateTime.plusHours( hours );

long minutes = tempDateTime.until( toDateTime, ChronoUnit.MINUTES );
tempDateTime = tempDateTime.plusMinutes( minutes );

long seconds = tempDateTime.until( toDateTime, ChronoUnit.SECONDS );

System.out.println( years + " years " + 
        months + " months " + 
        days + " days " +
        hours + " hours " +
        minutes + " minutes " +
        seconds + " seconds.");

//prints: 29 years 8 months 24 days 22 hours 54 minutes 50 seconds.

基本思想是这样的:创建一个临时的开始日期,并让整个年份结束。然后按年数调整该日期,使开始日期小于距结束一年。按降序对每个时间单位重复此操作。

最后是免责声明:我没有考虑不同的时区(两个日期应该在同一时区),我也没有测试/检查夏令时或日历中的其他变化(如萨摩亚的时区变化)如何影响此计算。因此,请谨慎使用。