One of the most common problems I see newbies to Objective-C and Cocoa struggle with on Stack Overflow is how to deal correctly with dates and times. Cocoa’s approach to date and time handling may indeed seem overly complex at first glance: where other languages’ standard libraries seem to get by with just one or two classes to cover this field, the Foundation framework employs a staggering array of separate classes:
NSTimeZone. These classes deal directly with date and time and you should be familiar with all of them. In addition, you should also understand the role of the
Let’s have a look at those classes one by one. As you will see, the Cocoa approach to date and time handling is not only quite easy to understand but also extremely flexible.
NSDate is the central class of the date/time handling in Foundation, and at the same time the simplest imaginable.
NSDate is nothing more than a wrapper around a single number: the number of seconds since 1 January, 2001, at 00:00 (midnight), UTC1. For values representing numbers of seconds, the framework uses a custom type,
NSTimeInterval, which is currently defined as a 64-bit floating point value. According to the documentation, this is enough to yield an impressive
sub-millisecond precision over a range of 10,000 years.
Represents an Absolute Point in Time
NSDate object always represents an absolute point in time.2 This insight has two important consequences:
There is no way to represent a certain date without including a specific time. For instance, to say that a particular
NSDateinstance represents 17 November 2011 makes no sense; you always have to include the particular time and time zone, such as 17 November 2011 00:00:00 +00:00 (or any other time of your choice).
If your app needs to store dates with less-than-second precision in order to represent entire days, months or years, you should either not use your own custom class for this or, better, define a rule how your app deals with the unused components of the date (e.g., set the time components of the date to 00:00:00 +00:00).
If you are sloppy and store dates with arbitrary time components, you will run into problems later when you want to compare or group multiple dates.
NSDatehas no concept of time zones. When it is midnight in London (17 November 2011 00:00:00 +00:00), it is only 6 pm on the day before in New York (16 November 2011 18:00:00 -06:00). Both dates represent the same point in time and are thus absolutely equal as far as
The implication of this is that you cannot store the time zone of a date and time in an
NSDateobject. If your app needs this information, you will have to store it in a different field. But more often than not, you will find that the time zone is actually not a field that should be stored with a date. Rather, it is a runtime preference of the person that is currently using your app, and your app should probably display most dates in the user’s current time zone.
How To Create An
NSDate That Represents A Specific Date?
The easiest way to create an
NSDate object is
[NSDate date];. This will return an instance that represents the current moment and is often useful in code when it comes to storing creation or modification dates of records or to measure certain time intervals in your app.
The more generic task of creating an instance that represents a specific date and time turns out to be not so straightforward. There is the
+dateWithTimeIntervalSinceReferenceDate: class method, but it requires you to know the interval in seconds between your desired date and the reference date (1 January 2001 00:00:00 +00:00). Turns out most people don’t count dates that way. That’s where the other classes come in.
Most people reading this will probably only ever use the same single calendar with its 12 months named January, February and so on, seven-day weeks, counting the years from the reputed birth of Jesus. It is easy to forget that (1) the current “western” Gregorian Calendar has only been introduced in 1582 and (2) there are many more calendars in practical use around the world today. The Foundation framework can currently handle ten different calendars3.
It should be clear that, to specify a date unambiguously, we need to specify the calendar we use. For instance, while today’s date falls into the year 2011 in the familiar Gregorian calendar, the current year is 2554 and 5772 in the Buddhist and Hebrew calendars, respectively.
In Cocoa, a calendar is represented by the
NSCalendar class. To create an instance of a specific calendar, pass one of the valid calendar identifiers to the initializer:
1 2 3
NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; NSCalendar *buddhist = [[NSCalendar alloc] initWithCalendarIdentifier:NSBuddhistCalendar]; NSCalendar *hebrew = [[NSCalendar alloc] initWithCalendarIdentifier:NSHebrewCalendar];
There are also two class methods,
+autoupdatingCurrentCalendar that return the current user’s preferred calendar. Note that the object returned by the latter method automatically adapts to changes in System Preferences.
The rest of the class is pretty straightforward. You can query the calendar for its configuration, i.e., things like the number of days that are in a month or which day is considered the first day of the week. Have a look at the documentation to get a feel for what is possible. There are also methods to split a date into its calendrical components or do the reverse but we are not quite ready to do that yet.
Any time specification is not precise enough without also indicating the time zone. I have already discussed that we need a way to reference time zones separately from
NSDate and the
NSTimeZone class does just that. There are several methods to create a time zone instance, the most straightforward being
Note, though, that the numeric offset from GMT is in many cases not enough to identify a specific time zone due to different daylight saving rules around the world. It is safer to specify a time zone by name using the
+timeZoneWithName: method. Valid names are of the form
+timeZoneWithAbbreviation: should be handled with care. It is supposed to create time zones from common abbreviations such as “PST” or “CEST”. The problem is that these abbreviations are not always unique – different countries might use the same abbreviation for different time zones or different abbreviations for the same time zone. You should avoid this ambiguity if possible.
Update November 22, 2011: In addition, the rules which time zone abbreviation is understood under a specific locale setting can change. As Cédric Luthi found out, Apple made a change in iOS 5 that causes the abbreviations “CET” and “CEST” (Central European (Summer) Time, very commonly used in Europe) to be no longer recognized if the user’s current locale setting is en_US. These abbreviations do still work with the en_GB locale, however. Another reason to avoid them completely if you ask me.
Last but not least, use the
+systemTimeZone method to get a reference to the user’s current time zone.
We have almost everything we need now to manipulate dates in our code. Our fourth class,
NSDateComponents, represents kind of the same information as
NSDate: a single point in time. Unlike the latter, however, an
NSDateComponents instance lets you access and manipulate every single calendrical component of that absolute point5, from the year down to the second and including such things as era, calendar, time zone and weekday.
Knowing this, let’s construct a date that represents the beginning of Steve Jobs’s Macworld 2007 keynote when he first introduced the iPhone:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; NSTimeZone *pacificTime = [NSTimeZone timeZoneWithName:@"America/Los_Angeles"]; NSDateComponents *dateComps = [[NSDateComponents alloc] init]; [dateComps setCalendar:gregorian]; [dateComps setYear:2007]; [dateComps setMonth:1]; [dateComps setDay:9]; [dateComps setTimeZone:pacificTime]; [dateComps setHour:9]; // keynote started at 9:00 am [dateComps setMinute:0]; // default value, can be omitted [dateComps setSecond:0]; // default value, can be omitted NSDate *dateOfKeynote = [dateComps date]; NSLog(@"Date of Keynote: %@", dateOfKeynote);
Date of Keynote: 2007-01-09 17:00:00 +0000
Hm, 17:00:00? But remember that
NSDate does not care about time zones. When printing an
NSLog(), the system always uses UTC, which is 8 hours ahead of San Francisco (or 7 hours during daylight savings time). So the resulting date is indeed correct.
Update November 28, 2011: Thanks to Shan for pointing out on Twitter that the code snippet above won’t work on OS X 10.6 or iOS 3.x because the
setTimeZone: methods are quite recent additions to
NSDateComponents. To stay compatible with the older OSs, create your
NSDateComponents instance just as above (without setting a calendar and time zone) and use the
dateFromComponents: method of
1 2 3 4
... [gregorian setTimeZone:pacificTime]; NSDate *dateOfKeynote = [gregorian dateFromComponents:dateComps]; NSLog(@"Date of Keynote: %@", dateOfKeynote);
You see, it’s just as easy.
Now that we have an
NSDateComponents instance, you would perhaps expect that you can get more information out of it. For example, let’s try to find out what day of the week the keynote was:
NSInteger weekday = [dateComps weekday]; // => -1 == NSUndefinedDateComponent
The documentation explains this:
An instance of
NSDateComponentsis not responsible for answering questions about a date beyond the information with which it was initialized. For example, if you initialize one with May 6, 2004, its weekday is
NSUndefinedDateComponent, not Thursday. To get the correct day of the week, you must create a suitable instance of
NSCalendar, create an
dateFromComponents:and then use
components:fromDate:to retrieve the weekday.
Let’s try that:
1 2 3
NSDate *dateOfKeynote = [dateComps date]; // or: [gregorian dateFromComponents:dateComps] NSDateComponents *weekdayComponents = [gregorian components:NSWeekdayCalendarUnit fromDate:dateOfKeynote]; NSInteger weekday = [weekdayComponents weekday]; // => 3 == Tuesday
Note how we can specify in the
-[NSCalendar components:fromDate:] method which date components we are interested in (using a bit mask). Some of the components can be expensive so it makes sense to only ask for the information you really need.
The combination of
NSCalendar is also the way to go for fancy date calculations. Say I want to create a date that goes back in time by exactly a month, a day and an hour from the current moment (using the current user’s calendar):
1 2 3 4 5 6 7
NSDate *now = [NSDate date]; NSDateComponents *comps = [[NSDateComponents alloc] init]; [comps setMonth:-1]; [comps setDay:-1]; [comps setHour:-1]; NSCalendar *calendar = [NSCalendar currentCalendar]; NSDate *newDate = [calendar dateByAddingComponents:comps toDate:now options:0];
NSDateComponents is an incredibly flexible und useful class. In combination with
NSCalendar, you can probably do all the date calculations you ever thought of.
Continued in Part 2: Date Parsing and Formatting
The classes I presented above give you a complete toolkit to work with date and time in your code. Two things are still missing, though: how to parse dates that come into your app as strings and how to output properly formatted dates as strings? Both of these tasks are handled by the
NSDateFormatter class, which I discuss in part 2 of this little series.
Implied in this is that
NSDateComponentsobjects are mutable whereas
NSDateinstances are immutable. ↩︎