在java.time中,如何计算添加月份的结果?

2022-09-01 23:27:28

在 JDK 8 中的 JSR-310 API 中,计算日期添加月份的结果的规则是什么?特别是,当您在1月31日等日期中添加1个月时会发生什么?java.time

LocalDate initial = LocalDate.of(2012, 1, 31);  // 31st January 2012
LocalDate result = initial.plusMonths(1);
// what is the result?

答案 1

简短的回答:

在此示例中,结果将是 2 月的最后一天 。2012-02-29

解释:

“如果你加上一个月,你会得到什么日期”这个问题,是一个可以解释的问题。为了避免这种情况,API 有一个明确的规则。结果将具有与输入相同的月中日,除非该日期无效,在这种情况下,结果是该月的最后一天。java.time

因此,1 月 31 日加 1 个月将产生 2 月 31 日,但由于这是一个无效的日期,因此结果是 2 月的最后有效日期,即 2 月 28 日还是 29 日,具体取决于它是闰年:

// normal case
2011-01-15 plus 1 month = 2011-02-15  // 15 Jan -> 15 Feb

// special rule choosing the last valid day-of-month
2011-01-31 plus 1 month = 2011-02-28  // 31 Jan -> 28 Feb (2011 is normal year)
2012-01-31 plus 1 month = 2012-02-29  // 31 Jan -> 29 Feb (2012 is leap year)

// same rule applies for months other than February
2013-03-31 plus 1 month = 2013-04-30  // 31 Mar -> 30 Apr (only 30 days in April)

无论添加一个月还是多个月,相同的规则都适用,并且始终基于生成的月份。即。首先添加月份(如有必要,调整年份),然后才考虑月份中的某一天。同样的规则也适用于减法。

// multiple months works on the month of the result
2013-10-31 plus 4 months = 2014-02-28   // last day of February
2013-10-31 minus 4 months = 2013-06-30  // last day of June

在日期中添加/减去年份时,相同的规则也适用 - 年份被添加,并且只有这样才能检查月份中的某一天在当月内的有效性。

// years use the same rule
2012-02-29 plus 1 year = 2013-02-28  // 29th February invalid so adjusted to 28th

如果业务逻辑需要不同的月份添加规则,最好的方法是编写一个或来打包您的特殊逻辑。TemporalAdjusterTemporalAmount


答案 2