Java Calendar.add给出了错误的一年

时间:2012-04-16 20:22:31

标签: java calendar

我的程序获取当前日期,然后在一个循环中,在该日期添加一周并打印出新日期。类似的东西:

Calendar cal = Calendar.getInstance();
for (int i=0; i < 52; i++) {
cal.add(Calendar.DATE, 7);
// print date out
}

add方法按照我预期的方式工作,直到它达到12月30日,此时年份从2012年跳到2013年。

所以,使用今天的4/16/2012日期,我测试了一些不同的输入:

this - cal.add(Calendar.DATE, 38*7);
yields- "date:1/7/2013"
this - cal.add(Calendar.DATE, 37*7);
yields- "date:12/31/2013"
this - cal.add(Calendar.DATE, 37*7-1);
yields- "date:12/30/2013"
this - cal.add(Calendar.DATE, 37*7-2);
yields- "date:12/29/2012"

所以我注意到年份是正确的,直到12月30日和12月31日,然后当它回到1月时它会再次纠正。这有什么原因吗?它与2012年是闰年有什么关系,或者我误解了添加方法

3 个答案:

答案 0 :(得分:2)

您是否使用SimpleDateFormat打印日期并使用YYYY生成年份?如果是这样,那就是问题所在。因为YYYY产生星期,而不是日历年。并且由于2012年12月30日是2013年日历的第1周,因此YYYY会产生2013年。要获取日历年,请在yyyy格式字符串中使用SimpleDateFormat

请参见https://bugs.openjdk.java.net/browse/JDK-8194625

答案 1 :(得分:1)

tl; dr

使用现代的java.time类,而不要使用可怕的旧类,例如Calendar

LocalDate                      // Represent a date-only value with `LocalDate`, without time-of-day and without time zone.
.now(                          // Capture the current date.
    ZoneId.systemDefault()     // Specify your desired/expected time zone explicitly.
)                              // Returns a `LocalDate` object.
.plusWeeks( 1 )                // Add a week, producing a new `LocalDate` object with values based on the original, per the immutable objects pattern.
.toString()                    // Generate text representing this date value in standard ISO 8601 format of YYYY-MM-DD.
  

2019-01-23

java.time

现代方法使用 java.time 类。

CalendarGregorianCalendar类很糟糕,设计得很糟糕,有缺陷。避免他们。现在专门由ZonedDateTime类代替。

LocalDate

LocalDate类表示没有日期,没有time zoneoffset-from-UTC的仅日期值。

时区对于确定日期至关重要。在任何给定时刻,日期都会在全球范围内变化。例如,Paris France午夜之后的几分钟是新的一天,而Montréal Québec仍然是“昨天”。

如果未指定时区,则JVM隐式应用其当前的默认时区。该默认值可能在运行时(!)期间change at any moment,因此您的结果可能会有所不同。最好将您的期望/期望时区明确指定为参数。

Continent/Region的格式指定proper time zone name,例如America/MontrealAfrica/CasablancaPacific/Auckland。切勿使用2-4个字母的缩写,例如ESTIST,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。

ZoneId z = ZoneId.of( "America/Montreal" ) ;  
LocalDate today = LocalDate.now( z ) ;

要生成标准ISO 8601格式的表示该日期值的文本,只需调用toString

String output = today.toString() ;

使用多种plus…minus…方法,日期数学很容易。

LocalDate weekLater = today.plusWeeks( 1 ) ;

您还可以将时间范围定义为PeriodDuration。然后添加它。

Period p = Period.ofWeeks( 1 ) ;
LocalDate weekLater = today.plus( p ) ;

您的示例

让我们测试一下示例日期。

LocalDate ld = LocalDate.of( 2012 , Month.APRIL , 16 ) ;

Period period38Weeks = Period.ofWeeks( 38 ) ;
Period period37Weeks = Period.ofWeeks( 37 ) ;
Period period37WeeksLess1Days = period37Weeks.minusDays( 1 ) ;
Period period37WeeksLess2Days = period37Weeks.minusDays( 2 ) ;

LocalDate later_38 = ld.plus( period38Weeks ) ;
LocalDate later_37 = ld.plus( period37Weeks ) ;
LocalDate later_37_1 = ld.plus( period37WeeksLess1Days ) ;
LocalDate later_37_2 = ld.plus( period37WeeksLess2Days ) ;

运行code live at IdeOne.com。没问题。第38周是2013年,而第37周是2012年。

  

later_38.toString():2013-01-07

     

later_37.toString():2012-12-31

     

later_37_1.toString():2012-12-30

     

later_37_2.toString():2012-12-29


关于 java.time

java.time框架已内置在Java 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.DateCalendarSimpleDateFormat

目前位于Joda-Timemaintenance mode项目建议迁移到java.time类。

要了解更多信息,请参见Oracle Tutorial。并在Stack Overflow中搜索许多示例和说明。规格为JSR 310

您可以直接与数据库交换 java.time 对象。使用符合JDBC driver或更高版本的JDBC 4.2。不需要字符串,不需要java.sql.*类。

在哪里获取java.time类?

ThreeTen-Extra项目使用其他类扩展了java.time。该项目为将来可能在java.time中添加内容提供了一个试验场。您可能会在这里找到一些有用的类,例如IntervalYearWeekYearQuartermore

答案 2 :(得分:0)

应该是:

cal.add(Calendar.DAY_OF_YEAR, 7);

Calendar.DATECalendar.DAY_OF_MONTH相同。