本地日期减去一个时间段得到错误结果

6

LocalDate减去一个Period(例如"28年1个月27天")会得到错误的结果。

但是,减去一个仅包含天数单位的Period(例如"10282"天)会得到正确的结果。有什么需要注意的吗?

public static void main(String[] args) {
    printAgeAndBirthday(1989, 2, 22);

    printBirthdayFromPeriod(28, 1, 27);
}
  private static void printBirthdayFromPeriod(int years, int months, int days) {
    final Period period = Period.of(years, months, days);
    final LocalDate now = LocalDate.now();
    final LocalDate birthday = now.minus(28, ChronoUnit.YEARS)
            .minus(1, ChronoUnit.MONTHS)
            .minus(27, ChronoUnit.DAYS);

    System.out.println("your birthday is : "+ birthday);//1989-02-19
    System.out.println("your birthday is : "+ now.minusYears(28).minusMonths(1).minusDays(27));//1989-02-19
    System.out.println("your birthday is : "+ now.minus(period));//1989-02-19
    System.out.println("your birthday is : "+period.subtractFrom(now));//1989-02-19
    System.out.println("your birthday is : "+ now.minus(Period.ofDays(10282)));//1989-02-22
}

private static void printAgeAndBirthday(int year, int month, int dayOfMonth) {
    LocalDate today = LocalDate.now();
    LocalDate birthday = LocalDate.of(year, month, dayOfMonth);

    Period p = Period.between(birthday, today);
    long p2 = ChronoUnit.DAYS.between(birthday, today);
    System.out.printf("You are %d years, %d months, and %d days old. (%d days total)%n",
            p.getYears(), p.getMonths(), p.getDays(), p2);

    LocalDate nextBDay = birthday.withYear(today.getYear());

    //If your birthday has occurred this year already, add 1 to the year.
    if (nextBDay.isBefore(today) || nextBDay.isEqual(today)) {
        nextBDay = nextBDay.plusYears(1);
    }

    Period p_1 = Period.between(today, nextBDay);
    long p_2 = ChronoUnit.DAYS.between(today, nextBDay);
    System.out.printf("There are %d months, and %d days until your next birthday. (%d total)%n",
            p_1.getMonths(), p_1.getDays(), p_2);
}

控制台日志:

You are 28 years, 1 months, and 27 days old. (10282 days total)
There are 10 months, and 4 days until your next birthday. (310 total)
your birthday is : 1989-02-19
your birthday is : 1989-02-19
your birthday is : 1989-02-19
your birthday is : 1989-02-19
your birthday is : 1989-02-22

java version : jdk1.8.0_45


4
不要使用 Period 进行精确计算。查看此相关帖子:https://dev59.com/7Z7ha4cB1Zd3GeqPnK80 - assylias
1
如果我从今天减去一个月零27天,那么得到的日期是2月19日。所以这是正确的。 - Holger
1个回答

14

你的情况可以简化为

LocalDate date1 = LocalDate.of(2017, 2, 22), date2 = LocalDate.of(2017, 4, 18);
Period p = Period.between(date1, date2);
System.out.println("date1 + p: "+date1.plus(p));
System.out.println("date2 - p: "+date2.minus(p));

这将会打印输出

date1 + p: 2017-04-18
date2 - p: 2017-02-19

换言之,年数无关紧要(除非涉及到一个闰年和另一个不是,但这里两个都不是)。以下是问题的示例:

February                       March                                                                                        April
19 20 21 22 23 24 25 26 27 28  1  2  3  4  5  6  7  8  9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31  1  2  3  4  5  6  7  8  9 10 11 12 13 14 15 16 17 18
 ↑        │                                                                                   ↑                                                                                ↑
 │        └──────────────────────────── plus one Month ───────────────────────────────────────┴───────────────────────── plus 27 days ─────────────────────────────────────────┤
 │                                                                                ↑                                                                                            ↓
 └───────────────────────── minus 27 days ────────────────────────────────────────┴─────────────────── minus one month ────────────────────────────────────────────────────────┘

如果您交换方向,则会发生更改:

Period p2 = Period.between(date2, date1);
System.out.println("date1 - p2: "+date1.minus(p2));
System.out.println("date2 + p2: "+date2.plus(p2));

将打印出来

date1 - p2: 2017-04-15
date2 + p2: 2017-02-22

因此,当你用年、月和日来表示一段时间时,方向变得很重要。相比之下,在两个日期之间的天数方面,纯数字是不变的:

LocalDate date1 = LocalDate.of(2017, 2, 22), date2 = LocalDate.of(2017, 4, 18);
Period p = Period.ofDays((int)ChronoUnit.DAYS.between(date1, date2));
System.out.println("date1 + p: "+date1.plus(p));
System.out.println("date2 - p: "+date2.minus(p));
date1 + p: 2017-04-18
date2 - p: 2017-02-22

非常感谢您的回答。我明白了。当我用Period减去LocalDate时,使用天作为单位是正确的。 - ChouChou
如果您在代码中使用以下日期:LocalDate date1 = LocalDate.of(1971, 1, 2),date2 = LocalDate.of(1972, 3, 1);您将会惊讶地发现以下结果: date1 + p: 1972-03-01 date2 - p: 1971-01-04预期date2 - p应该等于date1,但实际上并不是这样。显然,plus和minus函数不是反函数——这是一个错误吗? - Legna
@Legna,这个答案已经解释了对于日期来说加和减不是互逆函数,并且也解释了原因。 - Holger

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接