Java 8:计算两个LocalDateTime之间的差异

问题

我想计算两个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 seconds。我已经从这个website检查了我的结果(值为12/16/1984 07:45:5509/09/2014 19:46:45)。以下屏幕截图显示了输出:

Epoch Converter

我很确定月份值之后的字段从我的代码中出错了。任何建议都会非常有帮助。

##更新

我已经从另一个网站测试了我的结果,我得到的结果是不同的。这是:Calculate duration between two dates(结果: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.

从thislinkit应该是29 years 8 months 24 days 22 hours, 54 minutes and 50 seconds。因此算法也需要处理负数。
注意问题不在于哪个网站给了我什么结果,我需要知道正确的算法并且需要有正确的结果。


#1 热门回答(295 赞)

我发现最好的方法是使用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 热门回答(91 赞)

不幸的是,似乎没有一个时间段也可以跨越时间,因此你可能需要自己进行计算。

幸运的是,日期和时间类有很多实用方法可以在某种程度上简化它。这是一种计算差异的方法,虽然不一定是最快的:

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.

基本思路是这样的:创建一个临时的开始日期,并将整年打印到最后。然后按年数调整该日期,以便开始日期少于结束的一年。对每个时间单位按降序重复该操作。

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


#3 热门回答(13 赞)

这里有一个使用Duration和TimeUnit获得'hh:mm:ss'格式的例子。

Duration dur = Duration.between(LocalDateTimeIni, LocalDateTimeEnd);
                long millis = dur.toMillis();

                String.format("%02d:%02d:%02d", 
                        TimeUnit.MILLISECONDS.toHours(millis),
                        TimeUnit.MILLISECONDS.toMinutes(millis) - 
                        TimeUnit.HOURS.toMinutes(TimeUnit.MILLISECONDS.toHours(millis)),
                        TimeUnit.MILLISECONDS.toSeconds(millis) - 
                        TimeUnit.MINUTES.toSeconds(TimeUnit.MILLISECONDS.toMinutes(millis)));