时区日期处理(NSDate, NSCalendar, NSTimeZone)实例详解
参考:http://www.iliunian.com/1105.html
一、NSDate
NSDate存储的是世界标准时(UTC),输出时需要根据时区转换为本地时间, NSDate类提供了创建date,比较date以及计算两个date之间间隔的功能;Date对象是不可改变的。
如果你要创建date对象并表示当前日期,你可以alloc一个NSDate对象并调用init初始化。
NSDate *now = [[NSDate alloc] init];
或者使用NSDate的date类方法来创建一个日期对象。如果你需要与当前日期不同的日期,你可以使用NSDate的initWithTimeInterval或dateWithTimeInterval方法,你也可以使用更复杂的calendar或date components对象。
1)创建一定时间间隔的NSDate对象:
NSTimeInterval secondsPerDay = 24 * 60 * 60; NSDate *tomorrow = [[NSDate alloc] initWithTimeIntervalSinceNow:secondsPerDay]; NSDate *yesterday = [[NSDate alloc] initWithTimeIntervalSinceNow:-secondsPerDay]; [tomorrow release]; [yesterday release];
如果要对NSDate对象进行比较,可以使用isEqualToDate:, compare:, laterDate:和 earlierDate:方法。这些方法都进行精确比较,也就是说这些方法会一直精确比较到NSDate对象中秒一级。例如,你可能比较两个日期,如果他们之间的间隔在一分钟之内则认为这两个日期是相等的。在这种情况下使用,timeIntervalSinceDate:方法来对两个日期进行比较。下面的代码进行了示例:
if fabs(([date2 timeIntervalSinceDate:date1]) < 60)
二、NSCalendar & NSDateComponents
日历对象封装了对系统日期的计算,包括这一年开始,总天数以及划分。你将使用日历对象对绝对日期与date components(包括年,月,日,时,分,秒)进行转换。
NSCalendar定义了不同的日历,包括佛教历,格里高利历等(这些都与系统提供的本地化设置相关)。NSCalendar与NSDateComponents对象紧密相关。你可以通过NSCalendar对象的currentCalendar方法来获得当前系统用户设置的日历。
NSCalendar *currentCalendar = [NSCalendar currentCalendar]; NSCalendar *japaneseCalendar = [[NSCalendar alloc] initWithCalendarIdentifier:NSJapaneseCalendar]; NSCalendar *usersCalendar = [[NSLocale currentLocale] objectForKey:NSLocaleCalendar];
usersCalendar和currentCalendar对象是相等的,尽管他们是不同的对象。
你可以使用NSDateComponents对象来表示一个日期对象的组件——例如年,月,日和小时。如果要使一个NSDateComponents对象有意义,你必须将其与一个日历对象相关联。
NSDateComponents *components = [[NSDateComponents alloc] init]; [components setDay:6]; [components setMonth:5]; [components setYear:2004]; NSInteger weekday = [components weekday]; // Undefined (== NSUndefinedDateComponent)
要将一个日期对象解析到相应的date components,你可以使用NSCalendar的components:fromDate:方法。此外日期本身,你需要指定NSDateComponents对象返回组件。
NSDate *today = [NSDate date]; NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; NSDateComponents *weekdayComponents = [gregorian components:(NSDayCalendarUnit | NSWeekdayCalendarUnit) fromDate:today]; NSInteger day = [weekdayComponents day]; NSInteger weekday = [weekdayComponents weekday];
同样你也可以从NSDateComponents对象来创建NSDate对象:
NSDateComponents *components = [[NSDateComponents alloc] init]; [components setWeekday:2]; // Monday [components setWeekdayOrdinal:1]; // The first Monday in the month [components setMonth:5]; // May [components setYear:2008]; NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; NSDate *date = [gregorian dateFromComponents:components];
为了保证正确的行为,您必须确保使用的组件在日历上是有意义的。指定“出界”日历组件,如一个-6或2月30日在公历中的日期值产生未定义的行为。
你也可以创建一个不带年份的NSDate对象,这样的操作系统会自动生成一个年份,但在后面的代码中不会使用其自动生成的年份。
NSDateComponents *components = [[NSDateComponents alloc] init]; [components setMonth:11]; [components setDay:7]; NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; NSDate *birthday = [gregorian dateFromComponents:components];
下面的示例显示了如何从一个日历置换到另一个日历:
NSDateComponents *comps = [[NSDateComponents alloc] init]; [comps setDay:6]; [comps setMonth:5]; [comps setYear:2004]; NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; NSDate *date = [gregorian dateFromComponents:comps]; [comps release]; [gregorian release]; NSCalendar *hebrew = [[NSCalendar alloc] initWithCalendarIdentifier:NSHebrewCalendar]; NSUInteger unitFlags = NSDayCalendarUnit | NSMonthCalendarUnit | NSYearCalendarUnit; NSDateComponents *components = [hebrew components:unitFlags fromDate:date]; NSInteger day = [components day]; // 15 NSInteger month = [components month]; // 9 NSInteger year = [components year]; // 5764
历法计算,在当前时间加上一个半小时:
NSDate *today = [[NSDate alloc] init]; NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; NSDateComponents *offsetComponents = [[NSDateComponents alloc] init]; [offsetComponents setHour:1]; [offsetComponents setMinute:30]; // Calculate when, according to Tom Lehrer, World War III will end NSDate *endOfWorldWar3 = [gregorian dateByAddingComponents:offsetComponents toDate:today options:0];
获得当前星期中的星期天(使用格里高利历)
NSDate *today = [[NSDate alloc] init]; NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; // Get the weekday component of the current date NSDateComponents *weekdayComponents = [gregorian components:NSWeekdayCalendarUnit fromDate:today]; /* Create a date components to represent the number of days to subtract from the current date. The weekday value for Sunday in the Gregorian calendar is 1, so subtract 1 from the number of days to subtract from the date in question. (If today is Sunday, subtract 0 days.) */ NSDateComponents *componentsToSubtract = [[NSDateComponents alloc] init]; [componentsToSubtract setDay: 0 - ([weekdayComponents weekday] - 1)]; NSDate *beginningOfWeek = [gregorian dateByAddingComponents:componentsToSubtract toDate:today options:0]; /* Optional step: beginningOfWeek now has the same hour, minute, and second as the original date (today). To normalize to midnight, extract the year, month, and day components and create a new date from those components. */ NSDateComponents *components = [gregorian components:(NSYearCalendarUnit | NSMonthCalendarUnit | NSDayCalendarUnit) fromDate: beginningOfWeek]; beginningOfWeek = [gregorian dateFromComponents:components];
如何可以计算出一周的第一天(根据系统的日历设置):
NSDate *today = [[NSDate alloc] init]; NSDate *beginningOfWeek = nil; BOOL ok = [gregorian rangeOfUnit:NSWeekCalendarUnit startDate:&beginningOfWeek interval:NULL forDate: today]; 获得两个日期之间的间隔: NSDate *startDate = ...; NSDate *endDate = ...; NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar]; NSUInteger unitFlags = NSMonthCalendarUnit | NSDayCalendarUnit; NSDateComponents *components = [gregorian components:unitFlags fromDate:startDate toDate:endDate options:0]; NSInteger months = [components month]; NSInteger days = [components day];
使用Category来计算同一时代(AD|BC)两个日期午夜之间的天数:
@implementation NSCalendar (MySpecialCalculations)
-(NSInteger)daysWithinEraFromDate:(NSDate *) startDate toDate:(NSDate *) endDate {
NSInteger startDay=[self ordinalityOfUnit:NSDayCalendarUnit inUnit: NSEraCalendarUnit forDate:startDate];
NSInteger endDay=[self ordinalityOfUnit:NSDayCalendarUnit inUnit: NSEraCalendarUnit forDate:endDate];
return endDay-startDay;
}
@end
使用Category来计算不同时代(AD|BC)两个日期的天数:
@implementation NSCalendar (MyOtherMethod) -(NSInteger) daysFromDate:(NSDate *) startDate toDate:(NSDate *) endDate { NSCalendarUnit units=NSEraCalendarUnit | NSYearCalendarUnit | NSMonthCalendarUnit | NSDayCalendarUnit; NSDateComponents *comp1=[self components:units fromDate:startDate]; NSDateComponents *comp2=[self components:units fromDate endDate]; [comp1 setHour:12]; [comp2 setHour:12]; NSDate *date1=[self dateFromComponents: comp1]; NSDate *date2=[self dateFromComponents: comp2]; return [[self components:NSDayCalendarUnit fromDate:date1 toDate:date2 options:0] day]; } @end
判断一个日期是否在当前一周内(使用格里高利历):
-(BOOL)isDateThisWeek:(NSDate *)date { NSDate *start; NSTimeInterval extends; NSCalendar *cal=[NSCalendar autoupdatingCurrentCalendar]; NSDate *today=[NSDate date]; BOOL success= [cal rangeOfUnit:NSWeekCalendarUnit startDate:&start interval: &extends forDate:today]; if(!success) return NO; NSTimeInterval dateInSecs = [date timeIntervalSinceReferenceDate]; NSTimeInterval dayStartInSecs= [start timeIntervalSinceReferenceDate]; if(dateInSecs > dayStartInSecs && dateInSecs < (dayStartInSecs+extends)){ return YES; } else { return NO; } }
获取当前时间
NSDateFormatter*formatter = [[NSDateFormatteralloc] init]; [formatter setDateFormat:@"yyyy-MM-dd hh:mm:ss"]; NSString *locationString=[formatter stringFromDate: [NSDate date]];
另外的方法:
-(NSString *)getDate { NSDateFormatter*formatter = [[NSDateFormatteralloc] init]; [formatter setDateFormat:@"yyyy-MM-dd EEEE HH:mm:ss a"]; NSString *locationString=[formatter stringFromDate: [NSDate date]]; [formatter release]; return locationString; }
大写的H日期格式将默认为24小时制,小写的h日期格式将默认为12小时
不需要特别设置,只需要在dataFormat里设置类似”yyyy-MMM-dd”这样的格式就可以了
日期格式如下:
y 年 Year 1996; 96
M 年中的月份 Month July; Jul; 07
w 年中的周数 Number 27
W 月份中的周数 Number 2
D 年中的天数 Number 189
d 月份中的天数 Number 10
F 月份中的星期 Number 2
E 星期中的天数 Text Tuesday; Tue
a Am/pm 标记 Text PM
H 一天中的小时数(0-23) Number 0
k 一天中的小时数(1-24) Number 24
K am/pm 中的小时数(0-11) Number 0
h am/pm 中的小时数(1-12) Number 12
m 小时中的分钟数 Number 30
s 分钟中的秒数 Number 55
S 毫秒数 Number 978
z 时区 General time zone Pacific Standard Time; PST; GMT-08:00
Z 时区 RFC 822 time zone -0800
三、使用 NSTimeZone 取得世界各地时间的方法
NSLocale 区域对象封装了文化和语言约定,包括与时间相关的那些约定。
下列程式码将示范,如何利用 NSTimeZone 取得世界上已知的时区名称,并且透过这些名称来获得当地时间,如果在系统时间的取得上有任何疑问,可以参考取得 iOS 系统日期与星期的方法一文,其程式码如下。
//取得目前已知的所有地里名称 NSArray *timeZoneNames = [NSTimeZone knownTimeZoneNames]; //取得本地目前时间 NSDate *date = [NSDate date]; for(NSString *name in timeZoneNames) { NSTimeZone *timezone = [[NSTimeZone alloc] initWithName:name]; NSDateFormatter *formatter = [[NSDateFormatter alloc] init]; //设定时间格式 [formatter setDateFormat:@"YYYY-MM-d HH:mm:ss"]; //设定时区 [formatter setTimeZone:timezone]; //时间格式正规化并做时区校正 NSString *correctDate = [formatter stringFromDate:date]; NSLog(@"地点:%@ 当地时间:%@",[timezone name], correctDate); [formatter release]; [timezone release]; }
取得 iOS 系统日期与星期的方法
在之前的文章中已经说明如何取得 Device 里的 iOS 系统时间,在此将在示范如何使用 NSDate 取得系统的日期与星期,请看以下程式码。
NSDateFormatter *formatter = [[NSDateFormatter alloc] init]; NSDate *date = [NSDate date]; //正规化的格式设定 [formatter setDateFormat:@"YYYY-MM-dd' 'EEEE"]; //正规化取得的系统时间并显示 dateLabel.text = [formatter stringFromDate:date];
当然 NSFormatter 能正规化的格式不只这些,想知道其他的参数可以参考关于 NSDateFormatter 的二三事一文。
3. 取得 iOS 系统时间的方法
如何取得 Device 里的 iOS 系统时间,可以参考以下程式码。
NSDateFormatter *formatter = [[NSDateFormatter alloc] init]; NSDate *date = [NSDate date]; //正规化的格式设定 [formatter setTimeStyle:NSDateFormatterFullStyle]; //正规化取得的系统时间并显示 timeLabel.text = [formatter stringFromDate:date]; [formatter release];
在时间格式正规化的部份也有多种样式可供选择,其样式如下。
[formatter setTimeStyle:NSDateFormatterFullStyle]; [formatter setTimeStyle:NSDateFormatterLongStyle]; [formatter setTimeStyle:NSDateFormatterMediumStyle]; [formatter setTimeStyle:NSDateFormatterShortStyle];