There are two types of arithmetic in Noda Time: arithmetic on the time line (in some sense "absolute" arithmetic), and calendrical arithmetic. Noda Time deliberately separates the two, with the aim of avoiding subtle bugs where developers may be tempted to mix concepts inappropriately.
The Instant and
ZonedDateTime types both unambiguously
refer to a point in time (with the latter additionally holding time
zone and calendar information). We can add a length of time to that point to get
another point in time - but only if it's a truly fixed length of time, such as
"3 minutes". While a minute is a fixed length of time, a month isn't - so
the concept of adding "3 months" to an instant makes no sense. (Note
that Noda Time doesn't support leap seconds, otherwise even "3
minutes" wouldn't be a fixed length of time.)
These "fixed lengths of time" are represented in Noda Time with the
Duration type, and they can be
added to either an Instant or ZonedDateTime using either the + operator
or Plus methods:
Duration duration = Duration.FromMinutes(3);
Instant now = SystemClock.Instance.Now;
Instant future = now + duration; // Or now.Plus(duration)
ZonedDateTime nowInIsoUtc = now.InUtc();
ZonedDateTime thenInIsoUtc = nowInIsoUtc + duration;
There are also static methods (Add and Subtract), the - operator and
an instance method (Minus) on both Instant and ZonedDateTime.
Time line arithmetic is pretty simple, except you might not always get
what you expect when using ZonedDateTime, due to daylight saving transitions:
DateTimeZone london = DateTimeZoneProviders.Tzdb["Europe/London"];
// 12:45am on March 25th 2012
LocalDateTime local = new LocalDateTime(2012, 3, 25, 0, 45, 00);
ZonedDateTime before = london.AtStrictly(local);
ZonedDateTime after = before + Duration.FromMinutes(20);
We start off with a local time of 12.45am. The time that we add is effectively "experienced" time - as if we'd simply waited twenty minutes. However, at 1am on that day, the clocks in the Europe/London time zone go forward by an hour - so we end up with a local time of 2:05am, not the 1:05am you might have expected.
The reverse effect can happen at the other daylight saving transition, when clocks go backward
instead of forward: so twenty minutes after 1:45am could easily be 1:05am! So even though we
expose the concept of "local time" in ZonedDateTime, any arithmetic performed on it is computed
using the underlying time line.
The other type of arithmetic you'll typically want to perform with
Noda Time doesn't involve elapsed time directly, or even truly fixed
lengths of time - it involves calendrical concepts such as months.
These operations involve the "local" date and time types, which are
associated with a
CalendarSystem, but not a
time zone:
(LocalTime doesn't actually have an associated calendar, as Noda
Time assumes that all calendars model times in the same way, but
it's clearly closely related to the other two types.)
The simplest kind of arithmetic is adding some number of months,
years, days, hours etc to a value - a quantity of a single "unit",
in other words. This is easily achieved via the PlusXyz methods on
all of the local types:
LocalDate date = new LocalDate(2012, 2, 21);
date = date.PlusMonths(1); // March 21st 2012
date = date.PlusDays(-1); // March 20th 2012
LocalTime time = new LocalTime(7, 15, 0);
time = time.PlusHours(3); // 10:15 am
LocalDateTime dateTime = date + time;
dateTime = dateTime.PlusWeeks(1); // March 27th 2012, 10:15am
All of these types are immutable of course: the PlusXyz methods
don't modify the value they're called on; they return a new value
with the new date/time.
Even adding or subtracting a single unit can introduce problems due to unequal month lengths and the like. When adding a month or a year would create an invalid date, the day-of-month is truncated. For example:
LocalDate date = new LocalDate(2012, 2, 29);
LocalDate date1 = date.PlusYears(1); // February 28th 2013
LocalDate date2 = date.PlusMonths(1).PlusDays(1); // March 30th 2012
date2 = date2.PlusMonths(-1); // Back to February 29th 2012
LocalTime wraps around midnight transparently, but the same
operations on LocalDateTime will change the date appropriately too:
LocalTime time = new LocalTime(20, 30, 0);
time = time.PlusHours(6); // 2:30 am
LocalDateTime dateTime = new LocalDate(2012, 2, 21) + time;
dateTime = dateTime.PlusHours(-6); // 8:30pm on February 20th
Hopefully all of this is what you'd expect, but it's worth making sure the simple cases are clear before moving on to more complex ones.
PeriodSometimes you need a more general representation of the value to
add, which is where Period comes
in. This is essentially just a collection of unit/value pairs - so
you can have a period of "1 month and 3 days" or "2 weeks and 10
hours". Periods aren't normalized, so a period of "2 days" is not
the same as a period of "48 hours"; likewise if you ask for the
number of hours in a period of "1 day" the answer will be 0, not 24.
Single-unit periods can be obtained using the FromXyz methods, and
can then be added to the "local" types using the + operator or the
Plus method:
LocalDateTime dateTime = new LocalDateTime(2012, 2, 21, 7, 48, 0);
dateTime = dateTime + Period.FromDays(1) + Period.FromMinutes(1);
dateTime = dateTime.Plus(Period.FromHours(1));
Adding a period containing date units to a LocalTime or adding a
period containing time units to a LocalDate will result in an
ArgumentException.
Periods can be combined using simple addition too:
Period compound = Period.FromDays(1) + Period.FromMonths(1);
Again, this is very simple - the components in the two periods are simply summed, with no normalization. Subtraction works in the same way.
An alternative way of creating a period is to use PeriodBuilder
which is mutable, with a nullable property for each component:
Period compound = new PeriodBuilder { Days = 1, Months = 1 }.Build();
Adding a compound period can sometimes give unexpected results if you don't understand how they're handled, but the rule is extremely simple: One component is added at a time, starting with the most significant, and wrapping / truncating at each step.
It's easiest to think about where this can be confusing with an example. Suppose we add "one month minus three days" to January 30th 2011:
Period period = Period.FromMonths(1) - Period.FromDays(3);
LocalDate date = new LocalDate(2011, 1, 30);
date = date + period;
If you give this puzzle to a real person, they may well come up with an answer of "February 27th" by waiting until the last moment to check the validity. Noda Time will give an answer of February 25th, as the above code is effectively evaluated as:
Period period = Period.FromMonths(1) - Period.FromDays(3);
LocalDate date = new LocalDate(2011, 1, 30);
date = date + Period.FromMonths(1); // February 28th (truncated)
date = date - Period.FromDays(3); // February 25th
The benefit of this approach is simplicity and predictability: when you know the rules, it's very easy to work out what Noda Time will do. The downside is that if you don't know the rules, it looks like it's broken. It's possible that in a future version we'll implement a "smarter" API (as a separate option, probably, rather than replacing this one) - please drop a line to the mailing list if you have requirements in this area.
The opposite of this sort of arithmetic is answering questions such as "How many days are there until my birthday?" or "How many years, months, weeks, days, hours, minutes and seconds have I been married?"
Noda Time provides this functionality using Period.Between.
There are six overloads - two for each local type, with one using a
default set of units (year, month, day for dates; all time units for
times; year, month, day and all time units for date/times) and the
other allowing you to specify your own value. The units are specified
with the PeriodUnits enum, and
can be combined using the | operator. So for example, to find out
how many "months and days" old I am at the time of this writing, I'd use:
var birthday = new LocalDate(1976, 6, 19);
var today = new LocalDate(2012, 2, 21);
var period = Period.Between(birthday, today,
                            PeriodUnits.Months | PeriodUnits.Days);
Console.WriteLine("Age: {0} months and {1} days",
                  period.Months, period.Days);
Just as when adding periods, computing a period works from the largest specified unit down to the smallest, at each stage finding the appropriate component value with the greatest magnitude so that adding it to the running total doesn't "overshoot". This means that when computing a "positive" period (where the second argument is later than the first) every component value will be non-negative; when computing a "negative" period (where the second argument is earlier than the first) every component value will be zero or negative.
Note that these rules can very easily give asymmetric results. For example, consider:
// Remember that 2012 is a leap year...
var date1 = new LocalDate(2012, 2, 28);
var date2 = new LocalDate(2012, 3, 31);
var period1 = Period.Between(date1, date2);
var period2 = Period.Between(date2, date1);
Now period1 is "1 month and 3 days" - when we add a month to date1 we get to March 28th, and
then another 3 days takes us to March 31st. But period2 is "-1 month and -1 day" - when we subtract
a month from date2 we get to February 29th due to truncation, and then we only have to subtract
one more day to get to February 28th.
Again, this is easy to reason about and easy to implement. Contact the mailing list with extra requirements if you have them.
ZonedDateTime?All of this code using periods only works with the "local" types - notably there's
no part of the ZonedDateTime which mentions Period.
This is entirely deliberate, due to the complexities that time zones introduce. Every time
you perform a calculation on a ZonedDateTime, you may end up changing your offset from UTC.
So what does "12:30am plus one hour" mean when the hour between 1am and 2am is skipped?
Does it take us to 2:30am? That doesn't sound so bad - but when you add a year to a "midnight"
value and end up with 1am, that could be confusing... and there's even more confusion when you
think about ambiguous times. For example, supposed you have a ZonedDateTime representing
the first occurrence of 1:50am on a day when the clocks go back from 2am to 1am. What should
adding 20 minutes do? It could use the "elapsed" time, and end up with 1:10am (the second occurrence)
or it could end up with 2:10am (which would actually be 80 minutes later in elapsed time). None
of these options is particularly attractive.
Instead when you want to do calculations which aren't just based on a fixed number of ticks,
Noda Time forces you to convert to a local representation, perform all the arithmetic you want
there, then convert back to ZonedDateTime once, specifying how to handle ambiguous or
skipped times in the normal way. We believe this makes the API easier to follow and forces you
to think about the problems which you might otherwise brush under the carpet... but if you have
better suggestions, please raise them!
Currently Noda Time doesn't support arithmetic with OffsetDateTime
either, mostly because it's not clear what the use cases would be. You can always convert to either local or
zoned date/time values, perform arithmetic in that domain and convert back if necessary - but if you find
yourself in this situation, we'd love to hear about it on the mailing list.
One aspect of calendar arithmetic which is often required but doesn't fit in anywhere
else is finding an appropriate day of the week. LocalDateTime and LocalDate both provide
Next(IsoDayOfWeek) and Previous(IsoDayOfWeek) methods for this purpose. Both give the
strict next/previous date (or date/time) falling on the given day of the week - so calling
date.Next(IsoDayOfWeek.Sunday) on a date which is already on a Sunday will return a date
a week later, for example. The methods on LocalDateTime leave the time portion unchanged.
Currently all the calendars supported by Noda Time use the ISO week; if we ever support a calendar which doesn't, we'll see whether there's any need for a similar set of methods operating on non-ISO week days.
See also: