15

I'm not quite sure what field to use when adding more than 30 days to a Java Calendar object. Is there any difference in between Calendar.DAY_OF_MONTH and Calendar.DAY_OF_YEAR?

Example:

GregorianCalendar d = new GregorianCalendar();
d.add(Calendar.DAY_OF_YEAR, 90);

vs

GregorianCalendar d = new GregorianCalendar();
d.add(Calendar.DAY_OF_MONTH, 90);

Thanks.

Haes
  • 12,891
  • 11
  • 46
  • 50

2 Answers2

18

I don't think it makes a difference when you call add. The distinction is important when you call the getters.

Both methods work fine, right? For more than 30 days, as well as negative amounts.

The (admittedly complicated) source for GregorianCalendar#add has this section:

 case DAY_OF_MONTH: // synonym of DATE
 case DAY_OF_YEAR:
 case DAY_OF_WEEK:
    break;
Thilo
  • 257,207
  • 101
  • 511
  • 656
  • Thanks for the source confirmation. – Haes Mar 24 '10 at 07:59
  • 2
    The behaviour of over-/underflow when using the add method is well defined in the API documentation: "Overflow occurs when a field value exceeds its range and, as a result, the next larger field is incremented or decremented and the field value is adjusted back into its range." So overflowing the DAY_OF_MONTH field, means that the MONTH field (next larger) is incremented and the DAY_OF_MONTH field set to 1 (adjusted back into its valid range). – jarnbjo Mar 24 '10 at 10:40
  • @jarnbjo: and I think the exact same thing happens for DAY_OF_YEAR, DAY_OF_WEEK and DATE. – Thilo Mar 24 '10 at 10:51
  • @Thilo Is it the same for WEEK_OF_YEAR && WEEK_OF_MONTH in calender.add()?? Ans: Yes, It looks like same in source code "case WEEK_OF_YEAR: == case WEEK_OF_MONTH: == case DAY_OF_WEEK_IN_MONTH:" – Kanagavelu Sugumar Nov 09 '12 at 10:27
0

tl;dr

LocalDate.now( ZoneId.of( "America/Montreal" ) )
         .plusDays( 30 )

Details

Much easier now with the modern java.time classes that supplant the old Calendar & Date classes.

LocalDate

The LocalDate class represents a date-only value without time-of-day and without time zone.

A time zone is crucial in determining a date. For any given moment, the date varies around the globe by zone. For example, a few minutes after midnight in Paris France is a new day while still “yesterday” in Montréal Québec.

Specify a proper time zone name in the format of continent/region, such as America/Montreal, Africa/Casablanca, or Pacific/Auckland. Never use the 3-4 letter abbreviation such as EST or IST as they are not true time zones, not standardized, and not even unique(!).

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

You can add a number of days to that.

LocalDate later = today.plusDays( 30 );

Period

You can represent a span of time with the Period class.

Period thirtyDays = Period.ofDays( 30 );

You can perform date math by calling plus or minus methods.

LocalDate later = today.plus( thirtyDays );

About java.time

The java.time framework is built into Java 8 and later. These classes supplant the troublesome old legacy date-time classes such as java.util.Date, Calendar, & SimpleDateFormat.

The Joda-Time project, now in maintenance mode, advises migration to the java.time classes.

To learn more, see the Oracle Tutorial. And search Stack Overflow for many examples and explanations. Specification is JSR 310.

Where to obtain the java.time classes?

The ThreeTen-Extra project extends java.time with additional classes. This project is a proving ground for possible future additions to java.time. You may find some useful classes here such as Interval, YearWeek, YearQuarter, and more.

Community
  • 1
  • 1
Basil Bourque
  • 303,325
  • 100
  • 852
  • 1,154