TL;DR:使用payDate = payDate.plusDays(payDay - payDate.getDayOfMonth());
Calendar
您所描述的行为记录在 javadoc 中:
宽大
Calendar
有两种解释日历字段的模式,lenient和non-lenient。当 aCalendar
处于宽松模式时,它接受的日历字段值范围比它产生的范围更广。当 aCalendar
为 return by 重新计算日历字段值时get()
,所有日历字段都被规范化。例如,宽大的人将,GregorianCalendar
解释为 2 月 1 日。MONTH == JANUARY
DAY_OF_MONTH == 32
当 aCalendar
处于非宽松模式时,如果其日历字段中存在任何不一致,它将引发异常。例如,aGregorianCalendar
总是产生DAY_OF_MONTH
介于 1 和月份长度之间的值。如果设置了任何超出范围的字段值,则non-lenientGregorianCalendar
在计算其时间或日历字段值时会引发异常。
要显示此效果,请尝试将 a 的日期设置为Calendar
2020 年 1 月 70 日:
Calendar cal = Calendar.getInstance();
cal.clear();
cal.set(2020, Calendar.JANUARY, 70);
System.out.println(new SimpleDateFormat("yyyy-MM-dd").format(cal.getTime()));
输出
2020-03-10
如果你这样做,你会得到相同的结果:
cal.set(2020, Calendar.JANUARY, 1);
cal.add(Calendar.DAY_OF_MONTH, 69);
LocalDate
总是non-lenient,因此您不能将 day-of-month 值设置为超出范围的值。Calendar
但是,您可以通过将操作更改为“添加”而不是“设置”来获得与实际操作相同的结果。
因此,如果您有一个特定的日期,例如2020-01-29
问题中提到的日期,并且您想将日期值“设置”为 70 或 0,并且具有与之前相同的宽松溢出逻辑Calendar
,请执行以下操作:
LocalDate date = LocalDate.parse("2020-01-29");
date = date.plusDays(70 - date.getDayOfMonth());
System.out.println(date);
LocalDate date = LocalDate.parse("2020-01-29");
date = date.plusDays(0 - date.getDayOfMonth());
System.out.println(date);
输出
2020-03-10
2019-12-31
如您所见,date.plusDays(dayToSet - date.getDayOfMonth())
将为您提供所需的结果。