美文网首页iOS开发之中级iOS Developer
NSDate、NSTimeZone、NSCalendar和 NS

NSDate、NSTimeZone、NSCalendar和 NS

作者: asaBoat | 来源:发表于2017-05-19 10:53 被阅读75次

    相关资源

    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:https://img.haomeiwen.com/i3448645/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:https://img.haomeiwen.com/i3448645/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、NSCalendar和 NS

        本文链接:https://www.haomeiwen.com/subject/iufyxxtx.html