我的程序获取当前日期,然后在一个循环中,在该日期添加一周并打印出新日期。类似的东西:
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年是闰年有什么关系,或者我误解了添加方法
答案 0 :(得分:2)
您是否使用SimpleDateFormat
打印日期并使用YYYY
生成年份?如果是这样,那就是问题所在。因为YYYY
产生星期,而不是日历年。并且由于2012年12月30日是2013年日历的第1周,因此YYYY
会产生2013年。要获取日历年,请在yyyy
格式字符串中使用SimpleDateFormat
。
答案 1 :(得分:1)
使用现代的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 类。
Calendar
和GregorianCalendar
类很糟糕,设计得很糟糕,有缺陷。避免他们。现在专门由ZonedDateTime
类代替。
LocalDate
LocalDate
类表示没有日期,没有time zone或offset-from-UTC的仅日期值。
时区对于确定日期至关重要。在任何给定时刻,日期都会在全球范围内变化。例如,Paris France午夜之后的几分钟是新的一天,而Montréal Québec仍然是“昨天”。
如果未指定时区,则JVM隐式应用其当前的默认时区。该默认值可能在运行时(!)期间change at any moment,因此您的结果可能会有所不同。最好将您的期望/期望时区明确指定为参数。
以Continent/Region
的格式指定proper time zone name,例如America/Montreal
,Africa/Casablanca
或Pacific/Auckland
。切勿使用2-4个字母的缩写,例如EST
或IST
,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。
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 ) ;
您还可以将时间范围定义为Period
或Duration
。然后添加它。
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 8及更高版本中。这些类取代了麻烦的旧legacy日期时间类,例如java.util.Date
,Calendar
和SimpleDateFormat
。
目前位于Joda-Time的maintenance 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中添加内容提供了一个试验场。您可能会在这里找到一些有用的类,例如Interval
,YearWeek
,YearQuarter
和more。
答案 2 :(得分:0)
应该是:
cal.add(Calendar.DAY_OF_YEAR, 7);
Calendar.DATE
与Calendar.DAY_OF_MONTH
相同。