NSDate、NSTimeZone、NSCalendar和 NSDateFormatter

相关资源

cocoachina日历
本篇日历简单代码

NSDate 、NSTimeZone 、NSCalendar 和 NSDateFormatter

NSDate 、NSTimeZone 、NSCalendar和NSDateFormatter这四个类或者说对象都是相互独立的,他们之间有一定的关系,一定要认清。

  1. NSDate
// 获取现在时间,这个时间是现在的格林威治标准时间,和时区、语言都没有关系
[NSDate date];
  1. NSTimeZone
@property (class, readonly, copy) NSTimeZone *systemTimeZone;
@property (class, copy) NSTimeZone *defaultTimeZone;
@property (class, readonly, copy) NSTimeZone *localTimeZone

说明

  1. defaultTimeZone设置之前,和systemTimeZone共享一个timeZone对象
  2. defaultTimeZone设置之后,会影响NSDateFormatter和NSCalendar的默认时区(除非单独设置)。当你获取defaultTimeZone之后,获取的zone就是一个独立的对象了,即使你重新setDefaultTimeZone:也不会影响它。
  3. localTimeZone更像是一个单例对象(程序范围内),这个实例变量值会跟随defaultTimeZone的设置而变化。
  1. NSDateFormatter

// 时间戳转换成时间的方法(返回字符串)

  • (NSString *)timeStrWithInterval:(NSString *)interval {
    NSDate *date = [NSDate dateWithTimeIntervalSince1970:[interval doubleValue]/1000.0];
    NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
    formatter.timeZone = [NSTimeZone timeZoneWithName:@"Asia/Shanghai"];
    [formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
    NSString *timeString = [formatter stringFromDate:date];
    return timeString;
    }
 > 这里可以看到date格式化会受到时区影响,反向格式化也一样。

4.  NSCalendar
> 1. 日历,历法,一般历法都是遵循固定的规则的,具有周期性。日历都是已知的或可预测的。

> 2. NSDate是独立与任何历法的,它只是时间相对于某个时间点的时间差;NSDate是进行日历计算的基础。

> 3. 下面的两个例子(在**获取日历一周索引值**标题下),已经清晰反映了TimeZone和日历的关系。
> 通过[[NSCalendar currentCalendar] setTimeZone:[NSTimeZone localTimeZone]]对日历设置时区;


**其他重要API**

// 当前时间对应的星期几(索引)
[[NSCalendar currentCalendar] ordinalityOfUnit:NSCalendarUnitWeekday inUnit:NSCalendarUnitWeekOfMonth forDate:[NSDate date]];

// 当前时间对应的周是当前月中的第几周
[[NSCalendar currentCalendar] ordinalityOfUnit:NSCalendarUnitWeekOfMonth inUnit:NSCalendarUnitYear forDate:[NSDate date]];
[[NSCalendar currentCalendar] ordinalityOfUnit:NSCalendarUnitWeekOfMonth inUnit:NSCalendarUnitMonth forDate:[NSDate date]];

// 当前时间对应的周是当前年中的第几周
[[NSCalendar currentCalendar] ordinalityOfUnit:NSCalendarUnitWeekOfYear inUnit:NSCalendarUnitYear forDate:[NSDate date]];

#### 获取日历一周索引值

NSCalendar *calendar = [NSCalendar currentCalendar]; // 
NSLog(@"FirstWeekday = %ld",calendar.firstWeekday);//默认值 是 1。
[calendar setFirstWeekday:1]; //  设定日历每周的第一天从星期几开始,比如:如需设定从星期日开始(“开始”意味着日历的第一天(索引为1)对应的是星期天),则value传入1 ,如需设定从星期一开始,则value传入2 ,以此类推。


// 这里到天day和hour
NSDateComponents *comp = [calendar components:NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay | NSCalendarUnitHour fromDate:date];
NSLog(@"date = %@",date); // 2017-05-18 11:27:37 +0000
NSLog(@"dayComponet = %ld",comp.day);// 打印结果是18
NSLog(@"HourComponet = %ld",comp.hour);// 14H
// 从打印结果可以看出,calendar先把date加了3个小时,再处理component

// newDate
NSDate *newDate = [calendar dateFromComponents:comp];
NSLog(@"newDate = %@", newDate);// 2017-05-18 11:00:00 +0000 ,也就是说从calendar取出来的date是受NSTimeZone限制的。
// 从打印结果分析,calendar先components取出数值,再减去3小时。

// 设置为这个月的第一天
[comp setDay:1]; // 把comp的day属性设置为一号,即comp是5月1号14H
NSLog(@"dateComponet = %ld",comp.day);// 打印结果是1
NSDate *firstDayOfMonthDate = [calendar dateFromComponents:comp];
NSLog(@"firstDayOfMonthDate = %@",firstDayOfMonthDate);// 2017-05-01 11:00:00 +0000
NSDateFormatter *fommatter = [[NSDateFormatter alloc] init];
[fommatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
NSString *timeString = [fommatter stringFromDate:firstDayOfMonthDate];
NSLog(@"timeStr = %@",timeString);// 2017-05-01 14:00:00,这里可以看出fommatter也受时区影响。

// 经测试firstDayOfMonthDate 取值在2017-04-40 21:00:00 +0000 ~ 2017-05-01 21:00:00 +0000(不包括)之间,下面方法返回结果都一样,所以Calendar 受TimeZone影响,会给date自动加3小时。
NSUInteger firstWeekday = [calendar ordinalityOfUnit:NSCalendarUnitWeekday inUnit:NSCalendarUnitWeekOfMonth forDate:firstDayOfMonthDate]; // 这个月第一天在当前日历的顺序
// 返回某个特定时间(date)其对应的小的时间单元(smaller)在大的时间单元(larger)中的顺序位置(索引值)

// 如果这里值是2,因为前面设置的周日是第一位置,所以这里的索引2,对应的应该是周一。
NSLog(@"firstWeekday = %ld",firstWeekday);
> 本次代码分析的是component保留hour的情况,下面gif图是本人在第二天的运行的截图,上面代码分析是在18号。
> ![Calendar1.gif](http://upload-images.jianshu.io/upload_images/3448645-af67f4a9d587bd54.gif?imageMogr2/auto-orient/strip)


NSCalendar *calendar = [NSCalendar currentCalendar];
NSLog(@"FirstWeekday = %ld",calendar.firstWeekday);//默认值 是 1。
[calendar setFirstWeekday:2];

NSLog(@"date = %@",date); // 如果是5月31号21:00到6.1号21点(不包括)
// 这里只到天day
NSDateComponents *comp = [calendar components:NSCalendarUnitYear | NSCalendarUnitMonth | NSCalendarUnitDay fromDate:date];
NSLog(@"dateComponet = %ld",comp.day);// 打印结果是1,component 是6.1

// newDate
NSDate *newDate = [calendar dateFromComponents:comp];
NSLog(@"newDate = %@",newDate);// 这里的结果会是5月31号 21:00:00 +0000

// 设置为这个月的第一天
[comp setDay:1]; // 把comp的day属性设置为一号,即comp是6月1号
NSLog(@"dateComponet = %ld",comp.day);// 打印结果是1
NSDate *firstDayOfMonthDate = [calendar dateFromComponents:comp];
NSLog(@"date = %@",firstDayOfMonthDate);// 05-31 21:00:00 +0000

// 经测试firstDayOfMonthDate 取值在2017-04-40 21:00:00 +0000 ~ 2017-05-01 21:00:00 +0000(不包括)之间,下面方法返回结果都一样。date会加3个小时,变成6月份的日历。
NSUInteger firstWeekday = [calendar ordinalityOfUnit:NSCalendarUnitWeekday inUnit:NSCalendarUnitWeekOfMonth forDate:firstDayOfMonthDate]; // 这个月第一天在当前日历的顺序
// 返回某个特定时间(date)其对应的小的时间单元(smaller)在大的时间单元(larger)中的顺序位置(索引值)

// 这里值是1,因为前面设置的周一是第一位置,所以这里的索引1,对应的就是周一。
NSLog(@"firstWeekday = %ld",firstWeekday);

> 本次代码分析的是component只保留到day的情况 。下面gif图是本人在第二天的运行的截图,上面代码分析是在18号。
> ![Calendar.gif](http://upload-images.jianshu.io/upload_images/3448645-e90a6a4e3710ebb9.gif?imageMogr2/auto-orient/strip)

> 上面两个例子都没有问题,关键是
> 
> 1. 涉及到TimeZone,对TimeZone对象的分析很关键。[NSTimeZone setDefaultTimeZone:[NSTimeZone timeZoneWithName:@"Europe/Vilnius"]]的设置没在上面代码中显示,这是欧洲东时区三区的一个设置,也可以通过单独设置日历的时区对象属性。
> 2. 第二个关键部分是:找到对应月份和当前月对应的第一天,只要确保月份没有找错(component对应的月份就是最关键的月份),逻辑上就是对的。 如果date(格林威治标准时间GMT([NSDate date]))是5月31号21:00,component对应的数字就是6(month)和1(day),日历内部对应的“日期”就是6月1号00:00(以东三时区为例);component设置第一天对应的相关数值也是6(month)和1(day),然后用日历转化为date就是5月31号21:00,日历内部对应的“日期”就是6月1号00:00。

> 3. 第三个关键部分是:找到第一个“日期”对应的星期数,用的时间(NSDate)可能是上个月的时间,比如:date(格林威治标准时间GMT([NSDate date]))是5月31号21:00,然而时间对应“日期”(6月1号00:00)才是日历要查询的依据(日历会根据TimeZone自动调整date成“日期”)。

**找到日期对应月的第一天日期(更直接的方法)**
> 下面方法更直接,但是通过上面的代码,我们可以更好地分析本月第一天会是上个月的日期的情况。

  • (NSDate *)firstDayOfCurrentMonthForDate:(NSDate *)date {
    NSDate *startDate = nil;
    BOOL ok = [[NSCalendar currentCalendar] rangeOfUnit:NSMonthCalendarUnit startDate:&startDate interval:NULL forDate:date];
    return startDate;
    }

推荐阅读更多精彩内容

  • NSDate -- 表示一个绝对的时间点 NSTimeZone -- 时区信息 NSLocale -- 本地化信息...
    像小强一样活着阅读 2,409评论 6 16
  • iOS开发中,经常会遇到各种各样的时间问题,8小时时差,时间戳,求时间间隔,农历等等。解决办法网上比比皆是,但大多...
    小李龍彪阅读 1,906评论 1 3
  • 在iOS开发中,经常会遇到各种各样的时间问题,8小时时差,时间戳,求时间间隔,农历等等。解决办法网上比比皆是,但大...
    真巧了_嘿阅读 879评论 5 6
  • GitHub: https://github.com/GardenerYunEmail: gardeneryun@...
    园丁云阅读 3,226评论 3 38
  • 小乔初嫁周郎,三国厮杀正酣。青春败给烟火,宅男摔了键盘。
    王摩诘阅读 31评论 0 0